X-Git-Url: https://git.kernel.dk/?p=fio.git;a=blobdiff_plain;f=fio.c;h=5c45cae7f64103e0a8e30b16d7f704a2680f45e8;hp=44c8b3800fcc595afea8bca99c264a8fba202964;hb=0af7b542b69af5ab2400a9960a5bcde82c05723f;hpb=5289b84775b17eaf2a05481033761f100d5a01e2 diff --git a/fio.c b/fio.c index 44c8b380..5c45cae7 100644 --- a/fio.c +++ b/fio.c @@ -216,55 +216,6 @@ static void mark_random_map(struct thread_data *td, struct io_u *io_u) io_u->buflen = blocks * td->min_bs; } -static int get_next_offset(struct thread_data *td, unsigned long long *offset) -{ - unsigned long long b, rb; - long r; - - if (!td->sequential) { - unsigned long max_blocks = td->io_size / td->min_bs; - int loops = 50; - - do { - lrand48_r(&td->random_state, &r); - b = ((max_blocks - 1) * r / (RAND_MAX+1.0)); - rb = b + (td->file_offset / td->min_bs); - loops--; - } while (!random_map_free(td, rb) && loops); - - if (!loops) { - if (get_next_free_block(td, &b)) - return 1; - } - } else - b = td->last_bytes / td->min_bs; - - *offset = (b * td->min_bs) + td->file_offset; - if (*offset > td->file_size) - return 1; - - return 0; -} - -static unsigned int get_next_buflen(struct thread_data *td) -{ - unsigned int buflen; - long r; - - if (td->min_bs == td->max_bs) - buflen = td->min_bs; - else { - lrand48_r(&td->bsrange_state, &r); - buflen = (1 + (double) (td->max_bs - 1) * r / (RAND_MAX + 1.0)); - buflen = (buflen + td->min_bs - 1) & ~(td->min_bs - 1); - } - - if (buflen > td->io_size - td->this_io_bytes[td->ddir]) - buflen = td->io_size - td->this_io_bytes[td->ddir]; - - return buflen; -} - static inline void add_stat_sample(struct io_stat *is, unsigned long val) { if (val > is->max_val) @@ -327,6 +278,55 @@ static void add_bw_sample(struct thread_data *td, int ddir) td->stat_io_bytes[ddir] = td->this_io_bytes[ddir]; } +static int get_next_offset(struct thread_data *td, unsigned long long *offset) +{ + unsigned long long b, rb; + long r; + + if (!td->sequential) { + unsigned long max_blocks = td->io_size / td->min_bs; + int loops = 50; + + do { + lrand48_r(&td->random_state, &r); + b = ((max_blocks - 1) * r / (RAND_MAX+1.0)); + rb = b + (td->file_offset / td->min_bs); + loops--; + } while (!random_map_free(td, rb) && loops); + + if (!loops) { + if (get_next_free_block(td, &b)) + return 1; + } + } else + b = td->last_pos / td->min_bs; + + *offset = (b * td->min_bs) + td->file_offset; + if (*offset > td->real_file_size) + return 1; + + return 0; +} + +static unsigned int get_next_buflen(struct thread_data *td) +{ + unsigned int buflen; + long r; + + if (td->min_bs == td->max_bs) + buflen = td->min_bs; + else { + lrand48_r(&td->bsrange_state, &r); + buflen = (1 + (double) (td->max_bs - 1) * r / (RAND_MAX + 1.0)); + buflen = (buflen + td->min_bs - 1) & ~(td->min_bs - 1); + } + + if (buflen > td->io_size - td->this_io_bytes[td->ddir]) + buflen = td->io_size - td->this_io_bytes[td->ddir]; + + return buflen; +} + /* * busy looping version for the last few usec */ @@ -566,6 +566,19 @@ static void populate_io_u(struct thread_data *td, struct io_u *io_u) memcpy(io_u->buf, &hdr, sizeof(hdr)); } +static int td_io_prep(struct thread_data *td, struct io_u *io_u, int read) +{ + if (read) + io_u->ddir = DDIR_READ; + else + io_u->ddir = DDIR_WRITE; + + if (td->io_prep && td->io_prep(td, io_u)) + return 1; + + return 0; +} + static void put_io_u(struct thread_data *td, struct io_u *io_u) { list_del(&io_u->list); @@ -591,19 +604,6 @@ static struct io_u *__get_io_u(struct thread_data *td) return io_u; } -static int td_io_prep(struct thread_data *td, struct io_u *io_u, int read) -{ - if (read) - io_u->ddir = DDIR_READ; - else - io_u->ddir = DDIR_WRITE; - - if (td->io_prep && td->io_prep(td, io_u)) - return 1; - - return 0; -} - static struct io_u *get_io_u(struct thread_data *td) { struct io_u *io_u; @@ -612,6 +612,11 @@ static struct io_u *get_io_u(struct thread_data *td) if (!io_u) return NULL; + if (td->zone_bytes >= td->zone_size) { + td->zone_bytes = 0; + td->last_pos += td->zone_skip; + } + if (get_next_offset(td, &io_u->offset)) { put_io_u(td, io_u); return NULL; @@ -623,8 +628,8 @@ static struct io_u *get_io_u(struct thread_data *td) return NULL; } - if (io_u->buflen + io_u->offset > td->file_size) - io_u->buflen = td->file_size - io_u->offset; + if (io_u->buflen + io_u->offset > td->real_file_size) + io_u->buflen = td->real_file_size - io_u->offset; if (!io_u->buflen) { put_io_u(td, io_u); @@ -634,7 +639,7 @@ static struct io_u *get_io_u(struct thread_data *td) if (!td->sequential) mark_random_map(td, io_u); - td->last_bytes += io_u->buflen; + td->last_pos += io_u->buflen; if (td->verify != VERIFY_NONE) populate_io_u(td, io_u); @@ -751,11 +756,13 @@ static void io_completed(struct thread_data *td, struct io_u *io_u, gettimeofday(&e, NULL); if (!io_u->error) { + unsigned int bytes = io_u->buflen - io_u->resid; int idx = io_u->ddir; td->io_blocks[idx]++; - td->io_bytes[idx] += (io_u->buflen - io_u->resid); - td->this_io_bytes[idx] += (io_u->buflen - io_u->resid); + td->io_bytes[idx] += bytes; + td->zone_bytes += bytes; + td->this_io_bytes[idx] += bytes; msec = mtime_since(&io_u->issue_time, &e); @@ -765,7 +772,7 @@ static void io_completed(struct thread_data *td, struct io_u *io_u, if (td_write(td) && io_u->ddir == DDIR_WRITE) log_io_piece(td, io_u); - icd->bytes_done[idx] += (io_u->buflen - io_u->resid); + icd->bytes_done[idx] += bytes; } else icd->error = io_u->error; } @@ -1196,15 +1203,17 @@ static int file_size(struct thread_data *td) return 1; } - if (!td->file_size) - td->file_size = st.st_size; + td->real_file_size = st.st_size; + + if (!td->file_size || td->file_size > td->real_file_size) + td->file_size = td->real_file_size; return 0; } static int bdev_size(struct thread_data *td) { - size_t bytes; + unsigned long long bytes; int r; r = blockdev_size(td->fd, &bytes); @@ -1213,39 +1222,46 @@ static int bdev_size(struct thread_data *td) return 1; } + td->real_file_size = bytes; + /* * no extend possibilities, so limit size to device size if too large */ - if (!td->file_size || td->file_size > bytes) - td->file_size = bytes; + if (!td->file_size || td->file_size > td->real_file_size) + td->file_size = td->real_file_size; return 0; } static int get_file_size(struct thread_data *td) { - int ret; + int ret = 0; if (td->filetype == FIO_TYPE_FILE) ret = file_size(td); - else + else if (td->filetype == FIO_TYPE_BD) ret = bdev_size(td); + else + td->real_file_size = -1; if (ret) return ret; - if (td->file_offset > td->file_size) { - fprintf(stderr, "Client%d: offset larger than length (%Lu > %Lu)\n", td->thread_number, td->file_offset, td->file_size); + if (td->file_offset + td->file_size > td->real_file_size) { + fprintf(stderr, "Client%d: offset extends end (%Lu > %Lu)\n", td->thread_number, td->file_offset + td->file_size, td->real_file_size); return 1; } - td->io_size = td->file_size - td->file_offset; + td->io_size = td->file_size; if (td->io_size == 0) { fprintf(stderr, "Client%d: no io blocks\n", td->thread_number); td_verror(td, EINVAL); return 1; } + if (!td->zone_size) + td->zone_size = td->io_size; + td->total_io_size = td->io_size * td->loops; return 0; } @@ -1342,9 +1358,14 @@ static int setup_file(struct thread_data *td) if (td->odirect) flags |= O_DIRECT; - if (td_read(td)) - td->fd = open(td->file_name, flags | O_RDONLY); - else { + if (td_read(td)) { + if (td->filetype == FIO_TYPE_CHAR) + flags |= O_RDWR; + else + flags |= O_RDONLY; + + td->fd = open(td->file_name, flags); + } else { if (td->filetype == FIO_TYPE_FILE) { if (!td->overwrite) flags |= O_TRUNC; @@ -1585,17 +1606,23 @@ static void init_disk_util(struct thread_data *td) return; /* - * if this is inside a partition dir, jump back to parent + * for md/dm, there's no queue dir. we already have the right place */ - sprintf(tmp, "%s/queue", foo); + sprintf(tmp, "%s/stat", foo); if (stat(tmp, &st)) { - p = dirname(foo); - sprintf(tmp, "%s/queue", p); + /* + * if this is inside a partition dir, jump back to parent + */ + sprintf(tmp, "%s/queue", foo); if (stat(tmp, &st)) { - fprintf(stderr, "unknown sysfs layout\n"); - return; + p = dirname(foo); + sprintf(tmp, "%s/queue", p); + if (stat(tmp, &st)) { + fprintf(stderr, "unknown sysfs layout\n"); + return; + } + sprintf(foo, "%s", p); } - sprintf(foo, "%s", p); } disk_util_add(dev, foo); @@ -1613,9 +1640,10 @@ static void clear_io_state(struct thread_data *td) if (td->io_engine == FIO_SYNCIO) lseek(td->fd, SEEK_SET, 0); - td->last_bytes = 0; + td->last_pos = 0; td->stat_io_bytes[0] = td->stat_io_bytes[1] = 0; td->this_io_bytes[0] = td->this_io_bytes[1] = 0; + td->zone_bytes = 0; if (td->file_map) memset(td->file_map, 0, td->num_maps * sizeof(long)); @@ -1786,7 +1814,7 @@ static void show_ddir_status(struct thread_data *td, struct group_run_stats *rs, return; bw = td->io_bytes[ddir] / td->runtime[ddir]; - printf(" %s: io=%6luMiB, bw=%6luKiB/s, runt=%6lumsec\n", ddir_str[ddir], td->io_bytes[ddir] >> 20, bw, td->runtime[ddir]); + printf(" %s: io=%6lluMiB, bw=%6luKiB/s, runt=%6lumsec\n", ddir_str[ddir], td->io_bytes[ddir] >> 20, bw, td->runtime[ddir]); if (calc_lat(&td->slat_stat[ddir], &min, &max, &mean, &dev)) printf(" slat (msec): min=%5lu, max=%5lu, avg=%5.02f, dev=%5.02f\n", min, max, mean, dev); @@ -1922,6 +1950,9 @@ static void print_thread_status(void) if (td->verify) bytes_total += td->total_io_size; + if (td->zone_size && td->zone_skip) + bytes_total /= (td->zone_skip / td->zone_size); + bytes_done += td->io_bytes[DDIR_READ] +td->io_bytes[DDIR_WRITE]; check_str_update(td); @@ -2110,9 +2141,9 @@ static void show_group_stats(struct group_run_stats *rs, int id) printf("\nRun status group %d (all jobs):\n", id); if (rs->max_run[DDIR_READ]) - printf(" READ: io=%luMiB, aggrb=%lu, minb=%lu, maxb=%lu, mint=%lumsec, maxt=%lumsec\n", rs->io_mb[0], rs->agg[0], rs->min_bw[0], rs->max_bw[0], rs->min_run[0], rs->max_run[0]); + printf(" READ: io=%lluMiB, aggrb=%llu, minb=%llu, maxb=%llu, mint=%llumsec, maxt=%llumsec\n", rs->io_mb[0], rs->agg[0], rs->min_bw[0], rs->max_bw[0], rs->min_run[0], rs->max_run[0]); if (rs->max_run[DDIR_WRITE]) - printf(" WRITE: io=%luMiB, aggrb=%lu, minb=%lu, maxb=%lu, mint=%lumsec, maxt=%lumsec\n", rs->io_mb[1], rs->agg[1], rs->min_bw[1], rs->max_bw[1], rs->min_run[1], rs->max_run[1]); + printf(" WRITE: io=%lluMiB, aggrb=%llu, minb=%llu, maxb=%llu, mint=%llumsec, maxt=%llumsec\n", rs->io_mb[1], rs->agg[1], rs->min_bw[1], rs->max_bw[1], rs->min_run[1], rs->max_run[1]); } static void show_disk_util(void)