1 /*
2  * Copyright (C) 2010 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 /* A simple test of emmc random read and write performance.  When testing write
18  * performance, try it twice, once with O_SYNC compiled in, and once with it commented
19  * out.  Without O_SYNC, the close(2) blocks until all the dirty buffers are written
20  * out, but the numbers tend to be higher.
21  */
22 
23 #define _LARGEFILE64_SOURCE
24 #include <string.h>
25 #include <stdio.h>
26 #include <sys/types.h>
27 #include <sys/stat.h>
28 #include <fcntl.h>
29 #include <sys/time.h>
30 #include <stdlib.h>
31 #include <unistd.h>
32 #include <math.h>
33 
34 #define TST_BLK_SIZE 4096
35 /* Number of seconds to run the test */
36 #define TEST_LEN 10
37 
38 struct stats {
39     struct timeval start;
40     struct timeval end;
41     off64_t offset;
42 };
43 
44 static void usage(void) {
45         fprintf(stderr, "Usage: rand_emmc_perf [ -r | -w ] [-o] [-s count] [-f full_stats_filename] <size_in_mb> <block_dev>\n");
46         exit(1);
47 }
48 
49 static void print_stats(struct stats *stats_buf, int stats_count,
50                         char * full_stats_file)
51 {
52     int i;
53     struct timeval t;
54     struct timeval sum = { 0, 0 };
55     struct timeval max = { 0, 0 };
56     long long total_usecs;
57     long long avg_usecs;
58     long long max_usecs;
59     long long variance = 0;;
60     long long x;
61     double sdev;
62     FILE *full_stats = NULL;
63 
64     if (full_stats_file) {
65         full_stats = fopen(full_stats_file, "w");
66         if (full_stats == NULL) {
67             fprintf(stderr, "Cannot open full stats output file %s, ignoring\n",
68                     full_stats_file);
69         }
70     }
71 
72     for (i = 0; i < stats_count; i++) {
73         timersub(&stats_buf[i].end, &stats_buf[i].start, &t);
74         if (timercmp(&t, &max, >)) {
75             max = t;
76         }
77         if (full_stats) {
78             fprintf(full_stats, "%lld\n", (t.tv_sec * 1000000LL) + t.tv_usec);
79         }
80         timeradd(&sum, &t, &sum);
81     }
82 
83     if (full_stats) {
84         fclose(full_stats);
85     }
86 
87     max_usecs = (max.tv_sec * 1000000LL) + max.tv_usec;
88     total_usecs = (sum.tv_sec * 1000000LL) + sum.tv_usec;
89     avg_usecs = total_usecs / stats_count;
90     printf("average random %d byte iop time = %lld usecs\n",
91            TST_BLK_SIZE, avg_usecs);
92     printf("maximum random %d byte iop time = %lld usecs\n",
93            TST_BLK_SIZE, max_usecs);
94 
95     /* Now that we have the average (aka mean) go through the data
96      * again and compute the standard deviation.
97      * The formula is sqrt(sum_1_to_n((Xi - avg)^2)/n)
98      */
99     for (i = 0; i < stats_count; i++) {
100         timersub(&stats_buf[i].end, &stats_buf[i].start, &t);  /* Xi */
101         x = (t.tv_sec * 1000000LL) + t.tv_usec;                /* Convert to long long */
102         x = x - avg_usecs;                                     /* Xi - avg */
103         x = x * x;                                             /* (Xi - avg) ^ 2 */
104         variance += x;                                         /* Summation */
105     }
106     sdev = sqrt((double)variance/(double)stats_count);
107     printf("standard deviation of iops is %.2f\n", sdev);
108 }
109 
110 static void stats_test(int fd, int write_mode, off64_t max_blocks, int stats_count,
111                        char *full_stats_file)
112 {
113     struct stats *stats_buf;
114     char buf[TST_BLK_SIZE] = { 0 };
115     int i;
116 
117     stats_buf = malloc(stats_count * sizeof(struct stats));
118     if (stats_buf == NULL) {
119         fprintf(stderr, "Cannot allocate stats_buf\n");
120         exit(1);
121     }
122 
123     for (i = 0; i < stats_count; i++) {
124         gettimeofday(&stats_buf[i].start, NULL);
125 
126         if (lseek64(fd, (rand() % max_blocks) * TST_BLK_SIZE, SEEK_SET) < 0) {
127             fprintf(stderr, "lseek64 failed\n");
128         }
129 
130         if (write_mode) {
131             if (write(fd, buf, sizeof(buf)) != sizeof(buf)) {
132                 fprintf(stderr, "Short write\n");
133             }
134         } else {
135             if (read(fd, buf, sizeof(buf)) != sizeof(buf)) {
136                 fprintf(stderr, "Short read\n");
137             }
138         }
139 
140         gettimeofday(&stats_buf[i].end, NULL);
141     }
142 
143     print_stats(stats_buf, stats_count, full_stats_file);
144 }
145 
146 static void perf_test(int fd, int write_mode, off64_t max_blocks)
147 {
148     struct timeval start, end, res;
149     char buf[TST_BLK_SIZE] = { 0 };
150     long long iops = 0;
151     int msecs;
152 
153     res.tv_sec = 0;
154     gettimeofday(&start, NULL);
155     while (res.tv_sec < TEST_LEN) {
156         if (lseek64(fd, (rand() % max_blocks) * TST_BLK_SIZE, SEEK_SET) < 0) {
157             fprintf(stderr, "lseek64 failed\n");
158         }
159         if (write_mode) {
160             if (write(fd, buf, sizeof(buf)) != sizeof(buf)) {
161                 fprintf(stderr, "Short write\n");
162             }
163         } else {
164             if (read(fd, buf, sizeof(buf)) != sizeof(buf)) {
165                 fprintf(stderr, "Short read\n");
166             }
167         }
168         iops++;
169         gettimeofday(&end, NULL);
170         timersub(&end, &start, &res);
171     }
172     close(fd);
173 
174     /* The close can take a while when in write_mode as buffers are flushed.
175      * So get the time again. */
176     gettimeofday(&end, NULL);
177     timersub(&end, &start, &res);
178 
179     msecs = (res.tv_sec * 1000) + (res.tv_usec / 1000);
180     printf("%.0f %dbyte iops/sec\n", (float)iops * 1000 / msecs, TST_BLK_SIZE);
181 }
182 
183 int main(int argc, char *argv[])
184 {
185     int fd, fd2;
186     int write_mode = 0;
187     int o_sync = 0;
188     int stats_mode = 0;
189     int stats_count;
190     char *full_stats_file = NULL;
191     off64_t max_blocks;
192     unsigned int seed;
193     int c;
194 
195     while ((c = getopt(argc, argv, "+rwos:f:")) != -1) {
196         switch (c) {
197           case '?':
198           default:
199             usage();
200             break;
201 
202           case 'r':
203             /* Do nothing, read mode is the default */
204             break;
205 
206           case 'w':
207             write_mode = 1;
208             break;
209 
210           case 'o':
211             o_sync = O_SYNC;
212             break;
213 
214           case 's':
215             stats_mode = 1;
216             stats_count = atoi(optarg);
217             break;
218 
219           case 'f':
220             free(full_stats_file);
221             full_stats_file = strdup(optarg);
222             if (full_stats_file == NULL) {
223                 fprintf(stderr, "Cannot get full stats filename\n");
224             }
225             break;
226         }
227     }
228 
229     if (o_sync && !write_mode) {
230         /* Can only specify o_sync in write mode.  Probably doesn't matter,
231          * but clear o_sync if in read mode */
232         o_sync = 0;
233     }
234 
235     if ((argc - optind) != 2) {
236         usage();
237     }
238 
239     /* Size is given in megabytes, so compute the number of TST_BLK_SIZE blocks. */
240     max_blocks = atoll(argv[optind]) * ((1024*1024) / TST_BLK_SIZE);
241 
242     if ((fd = open(argv[optind + 1], O_RDWR | o_sync)) < 0) {
243         fprintf(stderr, "Cannot open block device %s\n", argv[optind + 1]);
244         exit(1);
245     }
246 
247     fd2 = open("/dev/urandom", O_RDONLY);
248     if (fd2 < 0) {
249         fprintf(stderr, "Cannot open /dev/urandom\n");
250     }
251     if (read(fd2, &seed, sizeof(seed)) != sizeof(seed)) {
252         fprintf(stderr, "Cannot read /dev/urandom\n");
253     }
254     close(fd2);
255     srand(seed);
256 
257     if (stats_mode) {
258         stats_test(fd, write_mode, max_blocks, stats_count, full_stats_file);
259     } else {
260         perf_test(fd, write_mode, max_blocks);
261     }
262     free(full_stats_file);
263 
264     exit(0);
265 }
266 
267