/**************************************************************************** * apps/system/cachespeed/cachespeed_main.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #include /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #define CACHESPEED_PREFIX "CACHE Speed: " #define REPEAT_NUM 1000 #ifdef CACHESPEED_PERFTIME #define TIME uint64_t #define CONVERT(cost) \ do \ { \ struct timespec ts; \ up_perf_convert(cost, &ts); \ cost = ts.tv_sec * 1000000000 + ts.tv_nsec; \ } while (0) #define TIMESTAMP(x) (x) = up_perf_gettime() #else #define TIME time_t #define CONVERT(cost) #define TIMESTAMP(x) \ do \ { \ struct timespec ts; \ clock_gettime(CLOCK_MONOTONIC, &ts); \ x = ts.tv_sec * 1000000000 + ts.tv_nsec; \ } while (0) #endif #define GET_DCACHE_LINE up_get_dcache_linesize() #define GET_ICACHE_LINE up_get_icache_linesize() #define GET_DCACHE_SIZE up_get_dcache_size() #define GET_ICACHE_SIZE up_get_icache_size() /**************************************************************************** * Private Types ****************************************************************************/ struct cachespeed_s { uintptr_t addr; size_t alloc; }; /**************************************************************************** * Private Data ****************************************************************************/ /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Name: setup ****************************************************************************/ static void setup(FAR struct cachespeed_s *cs) { struct mallinfo info = mallinfo(); /* Get the currently available memory from the system. We want the * memset range to be as large as possible in our tests to ensure * that the cache is filled with our dirty data */ cs->alloc = info.fordblks / 2; cs->addr = (uintptr_t)malloc(cs->alloc); if (cs->addr == 0) { printf(CACHESPEED_PREFIX "Unable to request memory.\n"); exit(EXIT_FAILURE); } /* Let's export the test message */ printf(CACHESPEED_PREFIX "address src: %" PRIxPTR "\n", cs->addr); } /**************************************************************************** * Name: teardown ****************************************************************************/ static void teardown(FAR struct cachespeed_s *cs) { free((void *)cs->addr); printf(CACHESPEED_PREFIX "Done!\n"); } /**************************************************************************** * Name: report_line ****************************************************************************/ static void report_line(size_t bytes, TIME cost) { double rate; /* There is a situation: if the time is 0, then the * calculated speed is wrong. */ CONVERT(cost); if (cost == 0) { printf(CACHESPEED_PREFIX "%d bytes cost time too small!\n", bytes); return; } /* rate = Test Data Size / Execution Time */ rate = 1.00 * bytes * REPEAT_NUM / cost; printf("%d Bytes: %4lf, %4llu, %4llu\n\r", bytes, rate, cost / REPEAT_NUM, cost); } /**************************************************************************** * Name: test_skeleton ****************************************************************************/ static void test_skeleton(FAR struct cachespeed_s *cs, const size_t cache_size, const size_t cache_line_size, int align, void (*func)(uintptr_t, uintptr_t), const char *name) { size_t update_size; printf("** %s [rate, avg, cost] in nanoseconds(bytes/nesc) %s **\n", name, align ? "align" : "unalign"); if (!align) { update_size = cache_line_size - 1; } else { update_size = cache_line_size; } for (size_t bytes = update_size; bytes <= cache_size; bytes = 2 * bytes) { irqstate_t irq; TIME start; TIME end; TIME cost = 0; /* Make sure that test with all the contents * of our address in the cache. */ up_flush_dcache_all(); irq = enter_critical_section(); for (int i = 0; i < REPEAT_NUM; i++) { memset((void *)cs->addr, 1, cs->alloc); TIMESTAMP(start); func(cs->addr, (uintptr_t)(cs->addr + bytes)); TIMESTAMP(end); cost += end - start; } leave_critical_section(irq); report_line(bytes, cost); } } /**************************************************************************** * Name: cachespeed_common ****************************************************************************/ static void cachespeed_common(struct cachespeed_s *cs) { test_skeleton(cs, GET_DCACHE_SIZE, GET_DCACHE_LINE, 1, up_invalidate_dcache, "dcache invalidate"); test_skeleton(cs, GET_DCACHE_SIZE, GET_DCACHE_LINE, 0, up_invalidate_dcache, "dcache invalidate"); test_skeleton(cs, GET_DCACHE_SIZE, GET_DCACHE_LINE, 1, up_clean_dcache, "dcache clean"); test_skeleton(cs, GET_DCACHE_SIZE, GET_DCACHE_LINE, 0, up_clean_dcache, "dcache clean"); test_skeleton(cs, GET_DCACHE_SIZE, GET_DCACHE_LINE, 1, up_flush_dcache, "dcache flush"); test_skeleton(cs, GET_DCACHE_SIZE, GET_DCACHE_LINE, 0, up_flush_dcache, "dcache flush"); test_skeleton(cs, GET_ICACHE_SIZE, GET_ICACHE_LINE, 1, up_invalidate_icache, "icache invalidate"); test_skeleton(cs, GET_ICACHE_SIZE, GET_ICACHE_LINE, 0, up_invalidate_icache, "icache invalidate"); } /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: cachespeed_main ****************************************************************************/ int main(int argc, FAR char *argv[]) { struct cachespeed_s cs = { .addr = 0, .alloc = 0 }; setup(&cs); cachespeed_common(&cs); teardown(&cs); return 0; }