| /* |
| * Copyright (C) 2011 The Android Open Source Project |
| * All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * * Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in |
| * the documentation and/or other materials provided with the |
| * distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS |
| * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE |
| * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, |
| * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, |
| * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS |
| * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED |
| * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, |
| * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT |
| * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF |
| * SUCH DAMAGE. |
| */ |
| |
| /* This program tries to benchmark stdio operations like fread() and |
| * fwrite() with various chunk sizes. We always read/write from /dev/zero |
| * to ensure that disk speed and caching don't change our results. |
| * |
| * We really do this to measure improvements in the low-level stdio |
| * features. |
| */ |
| |
| #include <stdio.h> |
| #include <time.h> |
| #include <string.h> |
| #include <errno.h> |
| |
| static char buffer[1024*1024]; |
| |
| /* Return current time in milli-seconds, as a double */ |
| static double |
| now_ms(void) |
| { |
| struct timespec ts; |
| clock_gettime(CLOCK_MONOTONIC, &ts); |
| return ts.tv_sec*1000. + ts.tv_nsec*1e-6; |
| } |
| |
| void read_file(FILE* fp, int chunkSize) |
| { |
| int totalSize = sizeof(buffer); |
| for ( ; totalSize > 0; totalSize -= chunkSize) { |
| fread(buffer, 1, chunkSize, fp); |
| } |
| } |
| |
| void write_file(FILE* fp, int chunkSize) |
| { |
| int totalSize = sizeof(buffer); |
| for ( ; totalSize > 0; totalSize -= chunkSize) { |
| fwrite(buffer, 1, chunkSize, fp); |
| } |
| } |
| |
| #define BENCH(op,...) \ |
| do { \ |
| double time_ms = now_ms(); \ |
| op ; \ |
| time_ms = now_ms() - time_ms; \ |
| double bandwidth = sizeof(buffer)*1000./1024./time_ms; \ |
| printf("bench %-30s %8.2f ms (%.1f KB/s) \n", #op, time_ms, bandwidth ); \ |
| } while (0) |
| |
| int main(void) |
| { |
| FILE* fp = fopen("/dev/zero", "rw"); |
| |
| if (fp == NULL) { |
| fprintf(stderr,"Could not open /dev/zero: %s\n", strerror(errno)); |
| return 1; |
| } |
| |
| BENCH(read_file(fp,1)); |
| BENCH(read_file(fp,2)); |
| BENCH(read_file(fp,3)); |
| BENCH(read_file(fp,4)); |
| BENCH(read_file(fp,8)); |
| BENCH(read_file(fp,16)); |
| BENCH(read_file(fp,32)); |
| BENCH(read_file(fp,64)); |
| BENCH(read_file(fp,256)); |
| BENCH(read_file(fp,1024)); |
| BENCH(read_file(fp,4096)); |
| BENCH(read_file(fp,16384)); |
| BENCH(read_file(fp,65536)); |
| |
| BENCH(write_file(fp,1)); |
| BENCH(write_file(fp,2)); |
| BENCH(write_file(fp,3)); |
| BENCH(write_file(fp,4)); |
| BENCH(write_file(fp,8)); |
| BENCH(write_file(fp,16)); |
| BENCH(write_file(fp,32)); |
| BENCH(write_file(fp,64)); |
| BENCH(write_file(fp,256)); |
| BENCH(write_file(fp,1024)); |
| BENCH(write_file(fp,4096)); |
| BENCH(write_file(fp,16384)); |
| BENCH(write_file(fp,65536)); |
| |
| fclose(fp); |
| return 0; |
| } |