1f17cfe81SBenoît Canet /* 2f17cfe81SBenoît Canet * Throttle infrastructure tests 3f17cfe81SBenoît Canet * 41fee955fSAlberto Garcia * Copyright Nodalink, EURL. 2013-2014 51fee955fSAlberto Garcia * Copyright Igalia, S.L. 2015 6f17cfe81SBenoît Canet * 7f17cfe81SBenoît Canet * Authors: 81fee955fSAlberto Garcia * Benoît Canet <benoit.canet@nodalink.com> 91fee955fSAlberto Garcia * Alberto Garcia <berto@igalia.com> 10f17cfe81SBenoît Canet * 11f17cfe81SBenoît Canet * This work is licensed under the terms of the GNU LGPL, version 2 or later. 12f17cfe81SBenoît Canet * See the COPYING.LIB file in the top-level directory. 13f17cfe81SBenoît Canet */ 14f17cfe81SBenoît Canet 15681c28a3SPeter Maydell #include "qemu/osdep.h" 16f17cfe81SBenoît Canet #include <glib.h> 17f17cfe81SBenoît Canet #include <math.h> 1813af91ebSStefan Hajnoczi #include "block/aio.h" 19da34e65cSMarkus Armbruster #include "qapi/error.h" 20f17cfe81SBenoît Canet #include "qemu/throttle.h" 212f78e491SChrysostomos Nanakos #include "qemu/error-report.h" 221fee955fSAlberto Garcia #include "block/throttle-groups.h" 23*31dce3ccSKevin Wolf #include "sysemu/block-backend.h" 24f17cfe81SBenoît Canet 25748bfb4eSStefan Weil static AioContext *ctx; 26748bfb4eSStefan Weil static LeakyBucket bkt; 27748bfb4eSStefan Weil static ThrottleConfig cfg; 28748bfb4eSStefan Weil static ThrottleState ts; 290e5b0a2dSBenoît Canet static ThrottleTimers tt; 30f17cfe81SBenoît Canet 3173f395faSStefan Weil /* useful function */ 32f17cfe81SBenoît Canet static bool double_cmp(double x, double y) 33f17cfe81SBenoît Canet { 34f17cfe81SBenoît Canet return fabsl(x - y) < 1e-6; 35f17cfe81SBenoît Canet } 36f17cfe81SBenoît Canet 37f17cfe81SBenoît Canet /* tests for single bucket operations */ 38f17cfe81SBenoît Canet static void test_leak_bucket(void) 39f17cfe81SBenoît Canet { 401588ab5dSAlberto Garcia throttle_config_init(&cfg); 411588ab5dSAlberto Garcia bkt = cfg.buckets[THROTTLE_BPS_TOTAL]; 421588ab5dSAlberto Garcia 43f17cfe81SBenoît Canet /* set initial value */ 44f17cfe81SBenoît Canet bkt.avg = 150; 45f17cfe81SBenoît Canet bkt.max = 15; 46f17cfe81SBenoît Canet bkt.level = 1.5; 47f17cfe81SBenoît Canet 48f17cfe81SBenoît Canet /* leak an op work of time */ 4913566fe3SStefan Hajnoczi throttle_leak_bucket(&bkt, NANOSECONDS_PER_SECOND / 150); 50f17cfe81SBenoît Canet g_assert(bkt.avg == 150); 51f17cfe81SBenoît Canet g_assert(bkt.max == 15); 52f17cfe81SBenoît Canet g_assert(double_cmp(bkt.level, 0.5)); 53f17cfe81SBenoît Canet 54f17cfe81SBenoît Canet /* leak again emptying the bucket */ 5513566fe3SStefan Hajnoczi throttle_leak_bucket(&bkt, NANOSECONDS_PER_SECOND / 150); 56f17cfe81SBenoît Canet g_assert(bkt.avg == 150); 57f17cfe81SBenoît Canet g_assert(bkt.max == 15); 58f17cfe81SBenoît Canet g_assert(double_cmp(bkt.level, 0)); 59f17cfe81SBenoît Canet 60f17cfe81SBenoît Canet /* check that the bucket level won't go lower */ 6113566fe3SStefan Hajnoczi throttle_leak_bucket(&bkt, NANOSECONDS_PER_SECOND / 150); 62f17cfe81SBenoît Canet g_assert(bkt.avg == 150); 63f17cfe81SBenoît Canet g_assert(bkt.max == 15); 64f17cfe81SBenoît Canet g_assert(double_cmp(bkt.level, 0)); 65eb8a1a1cSAlberto Garcia 66eb8a1a1cSAlberto Garcia /* check that burst_level leaks correctly */ 67eb8a1a1cSAlberto Garcia bkt.burst_level = 6; 68eb8a1a1cSAlberto Garcia bkt.max = 250; 69eb8a1a1cSAlberto Garcia bkt.burst_length = 2; /* otherwise burst_level will not leak */ 70eb8a1a1cSAlberto Garcia throttle_leak_bucket(&bkt, NANOSECONDS_PER_SECOND / 100); 71eb8a1a1cSAlberto Garcia g_assert(double_cmp(bkt.burst_level, 3.5)); 72eb8a1a1cSAlberto Garcia 73eb8a1a1cSAlberto Garcia throttle_leak_bucket(&bkt, NANOSECONDS_PER_SECOND / 100); 74eb8a1a1cSAlberto Garcia g_assert(double_cmp(bkt.burst_level, 1)); 75eb8a1a1cSAlberto Garcia 76eb8a1a1cSAlberto Garcia throttle_leak_bucket(&bkt, NANOSECONDS_PER_SECOND / 100); 77eb8a1a1cSAlberto Garcia g_assert(double_cmp(bkt.burst_level, 0)); 78eb8a1a1cSAlberto Garcia 79eb8a1a1cSAlberto Garcia throttle_leak_bucket(&bkt, NANOSECONDS_PER_SECOND / 100); 80eb8a1a1cSAlberto Garcia g_assert(double_cmp(bkt.burst_level, 0)); 81f17cfe81SBenoît Canet } 82f17cfe81SBenoît Canet 83f17cfe81SBenoît Canet static void test_compute_wait(void) 84f17cfe81SBenoît Canet { 85f9d05885SAlberto Garcia unsigned i; 86f17cfe81SBenoît Canet int64_t wait; 87f17cfe81SBenoît Canet int64_t result; 88f17cfe81SBenoît Canet 891588ab5dSAlberto Garcia throttle_config_init(&cfg); 901588ab5dSAlberto Garcia bkt = cfg.buckets[THROTTLE_BPS_TOTAL]; 911588ab5dSAlberto Garcia 92f17cfe81SBenoît Canet /* no operation limit set */ 93f17cfe81SBenoît Canet bkt.avg = 0; 94f17cfe81SBenoît Canet bkt.max = 15; 95f17cfe81SBenoît Canet bkt.level = 1.5; 96f17cfe81SBenoît Canet wait = throttle_compute_wait(&bkt); 97f17cfe81SBenoît Canet g_assert(!wait); 98f17cfe81SBenoît Canet 99f17cfe81SBenoît Canet /* zero delta */ 100f17cfe81SBenoît Canet bkt.avg = 150; 101f17cfe81SBenoît Canet bkt.max = 15; 102f17cfe81SBenoît Canet bkt.level = 15; 103f17cfe81SBenoît Canet wait = throttle_compute_wait(&bkt); 104f17cfe81SBenoît Canet g_assert(!wait); 105f17cfe81SBenoît Canet 106f17cfe81SBenoît Canet /* below zero delta */ 107f17cfe81SBenoît Canet bkt.avg = 150; 108f17cfe81SBenoît Canet bkt.max = 15; 109f17cfe81SBenoît Canet bkt.level = 9; 110f17cfe81SBenoît Canet wait = throttle_compute_wait(&bkt); 111f17cfe81SBenoît Canet g_assert(!wait); 112f17cfe81SBenoît Canet 113f17cfe81SBenoît Canet /* half an operation above max */ 114f17cfe81SBenoît Canet bkt.avg = 150; 115f17cfe81SBenoît Canet bkt.max = 15; 116f17cfe81SBenoît Canet bkt.level = 15.5; 117f17cfe81SBenoît Canet wait = throttle_compute_wait(&bkt); 118f17cfe81SBenoît Canet /* time required to do half an operation */ 11913566fe3SStefan Hajnoczi result = (int64_t) NANOSECONDS_PER_SECOND / 150 / 2; 120f17cfe81SBenoît Canet g_assert(wait == result); 121f9d05885SAlberto Garcia 122f9d05885SAlberto Garcia /* Perform I/O for 2.2 seconds at a rate of bkt.max */ 123f9d05885SAlberto Garcia bkt.burst_length = 2; 124f9d05885SAlberto Garcia bkt.level = 0; 125f9d05885SAlberto Garcia bkt.avg = 10; 126f9d05885SAlberto Garcia bkt.max = 200; 127f9d05885SAlberto Garcia for (i = 0; i < 22; i++) { 128f9d05885SAlberto Garcia double units = bkt.max / 10; 129f9d05885SAlberto Garcia bkt.level += units; 130f9d05885SAlberto Garcia bkt.burst_level += units; 131f9d05885SAlberto Garcia throttle_leak_bucket(&bkt, NANOSECONDS_PER_SECOND / 10); 132f9d05885SAlberto Garcia wait = throttle_compute_wait(&bkt); 133f9d05885SAlberto Garcia g_assert(double_cmp(bkt.burst_level, 0)); 134f9d05885SAlberto Garcia g_assert(double_cmp(bkt.level, (i + 1) * (bkt.max - bkt.avg) / 10)); 135f9d05885SAlberto Garcia /* We can do bursts for the 2 seconds we have configured in 136f9d05885SAlberto Garcia * burst_length. We have 100 extra miliseconds of burst 137f9d05885SAlberto Garcia * because bkt.level has been leaking during this time. 138f9d05885SAlberto Garcia * After that, we have to wait. */ 139f9d05885SAlberto Garcia result = i < 21 ? 0 : 1.8 * NANOSECONDS_PER_SECOND; 140f9d05885SAlberto Garcia g_assert(wait == result); 141f9d05885SAlberto Garcia } 142f17cfe81SBenoît Canet } 143f17cfe81SBenoît Canet 144f17cfe81SBenoît Canet /* functions to test ThrottleState initialization/destroy methods */ 145f17cfe81SBenoît Canet static void read_timer_cb(void *opaque) 146f17cfe81SBenoît Canet { 147f17cfe81SBenoît Canet } 148f17cfe81SBenoît Canet 149f17cfe81SBenoît Canet static void write_timer_cb(void *opaque) 150f17cfe81SBenoît Canet { 151f17cfe81SBenoît Canet } 152f17cfe81SBenoît Canet 153f17cfe81SBenoît Canet static void test_init(void) 154f17cfe81SBenoît Canet { 155f17cfe81SBenoît Canet int i; 156f17cfe81SBenoît Canet 1570e5b0a2dSBenoît Canet /* fill the structures with crap */ 158f17cfe81SBenoît Canet memset(&ts, 1, sizeof(ts)); 1590e5b0a2dSBenoît Canet memset(&tt, 1, sizeof(tt)); 160f17cfe81SBenoît Canet 1610e5b0a2dSBenoît Canet /* init structures */ 1620e5b0a2dSBenoît Canet throttle_init(&ts); 1630e5b0a2dSBenoît Canet throttle_timers_init(&tt, ctx, QEMU_CLOCK_VIRTUAL, 16413af91ebSStefan Hajnoczi read_timer_cb, write_timer_cb, &ts); 165f17cfe81SBenoît Canet 166f17cfe81SBenoît Canet /* check initialized fields */ 1670e5b0a2dSBenoît Canet g_assert(tt.clock_type == QEMU_CLOCK_VIRTUAL); 1680e5b0a2dSBenoît Canet g_assert(tt.timers[0]); 1690e5b0a2dSBenoît Canet g_assert(tt.timers[1]); 170f17cfe81SBenoît Canet 171f17cfe81SBenoît Canet /* check other fields where cleared */ 172f17cfe81SBenoît Canet g_assert(!ts.previous_leak); 173f17cfe81SBenoît Canet g_assert(!ts.cfg.op_size); 174f17cfe81SBenoît Canet for (i = 0; i < BUCKETS_COUNT; i++) { 175f17cfe81SBenoît Canet g_assert(!ts.cfg.buckets[i].avg); 176f17cfe81SBenoît Canet g_assert(!ts.cfg.buckets[i].max); 177f17cfe81SBenoît Canet g_assert(!ts.cfg.buckets[i].level); 178f17cfe81SBenoît Canet } 179f17cfe81SBenoît Canet 1800e5b0a2dSBenoît Canet throttle_timers_destroy(&tt); 181f17cfe81SBenoît Canet } 182f17cfe81SBenoît Canet 183f17cfe81SBenoît Canet static void test_destroy(void) 184f17cfe81SBenoît Canet { 185f17cfe81SBenoît Canet int i; 1860e5b0a2dSBenoît Canet throttle_init(&ts); 1870e5b0a2dSBenoît Canet throttle_timers_init(&tt, ctx, QEMU_CLOCK_VIRTUAL, 18813af91ebSStefan Hajnoczi read_timer_cb, write_timer_cb, &ts); 1890e5b0a2dSBenoît Canet throttle_timers_destroy(&tt); 190f17cfe81SBenoît Canet for (i = 0; i < 2; i++) { 1910e5b0a2dSBenoît Canet g_assert(!tt.timers[i]); 192f17cfe81SBenoît Canet } 193f17cfe81SBenoît Canet } 194f17cfe81SBenoît Canet 195f17cfe81SBenoît Canet /* function to test throttle_config and throttle_get_config */ 196f17cfe81SBenoît Canet static void test_config_functions(void) 197f17cfe81SBenoît Canet { 198f17cfe81SBenoît Canet int i; 199f17cfe81SBenoît Canet ThrottleConfig orig_cfg, final_cfg; 200f17cfe81SBenoît Canet 201f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_BPS_TOTAL].avg = 153; 202f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_BPS_READ].avg = 56; 203f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_BPS_WRITE].avg = 1; 204f17cfe81SBenoît Canet 205f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_OPS_TOTAL].avg = 150; 206f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_OPS_READ].avg = 69; 207f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_OPS_WRITE].avg = 23; 208f17cfe81SBenoît Canet 209f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_BPS_TOTAL].max = 0; /* should be corrected */ 210f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_BPS_READ].max = 1; /* should not be corrected */ 211f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_BPS_WRITE].max = 120; 212f17cfe81SBenoît Canet 213f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_OPS_TOTAL].max = 150; 214f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_OPS_READ].max = 400; 215f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_OPS_WRITE].max = 500; 216f17cfe81SBenoît Canet 217f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_BPS_TOTAL].level = 45; 218f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_BPS_READ].level = 65; 219f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_BPS_WRITE].level = 23; 220f17cfe81SBenoît Canet 221f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_OPS_TOTAL].level = 1; 222f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_OPS_READ].level = 90; 223f17cfe81SBenoît Canet orig_cfg.buckets[THROTTLE_OPS_WRITE].level = 75; 224f17cfe81SBenoît Canet 225f17cfe81SBenoît Canet orig_cfg.op_size = 1; 226f17cfe81SBenoît Canet 2270e5b0a2dSBenoît Canet throttle_init(&ts); 2280e5b0a2dSBenoît Canet throttle_timers_init(&tt, ctx, QEMU_CLOCK_VIRTUAL, 22913af91ebSStefan Hajnoczi read_timer_cb, write_timer_cb, &ts); 230f17cfe81SBenoît Canet /* structure reset by throttle_init previous_leak should be null */ 231f17cfe81SBenoît Canet g_assert(!ts.previous_leak); 2320e5b0a2dSBenoît Canet throttle_config(&ts, &tt, &orig_cfg); 233f17cfe81SBenoît Canet 234f17cfe81SBenoît Canet /* has previous leak been initialized by throttle_config ? */ 235f17cfe81SBenoît Canet g_assert(ts.previous_leak); 236f17cfe81SBenoît Canet 237f17cfe81SBenoît Canet /* get back the fixed configuration */ 238f17cfe81SBenoît Canet throttle_get_config(&ts, &final_cfg); 239f17cfe81SBenoît Canet 2400e5b0a2dSBenoît Canet throttle_timers_destroy(&tt); 241f17cfe81SBenoît Canet 242f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_BPS_TOTAL].avg == 153); 243f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_BPS_READ].avg == 56); 244f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_BPS_WRITE].avg == 1); 245f17cfe81SBenoît Canet 246f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_OPS_TOTAL].avg == 150); 247f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_OPS_READ].avg == 69); 248f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_OPS_WRITE].avg == 23); 249f17cfe81SBenoît Canet 250f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_BPS_TOTAL].max == 15.3);/* fixed */ 251f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_BPS_READ].max == 1); /* not fixed */ 252f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_BPS_WRITE].max == 120); 253f17cfe81SBenoît Canet 254f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_OPS_TOTAL].max == 150); 255f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_OPS_READ].max == 400); 256f17cfe81SBenoît Canet g_assert(final_cfg.buckets[THROTTLE_OPS_WRITE].max == 500); 257f17cfe81SBenoît Canet 258f17cfe81SBenoît Canet g_assert(final_cfg.op_size == 1); 259f17cfe81SBenoît Canet 260f17cfe81SBenoît Canet /* check bucket have been cleared */ 261f17cfe81SBenoît Canet for (i = 0; i < BUCKETS_COUNT; i++) { 262f17cfe81SBenoît Canet g_assert(!final_cfg.buckets[i].level); 263f17cfe81SBenoît Canet } 264f17cfe81SBenoît Canet } 265f17cfe81SBenoît Canet 266f17cfe81SBenoît Canet /* functions to test is throttle is enabled by a config */ 267f17cfe81SBenoît Canet static void set_cfg_value(bool is_max, int index, int value) 268f17cfe81SBenoît Canet { 269f17cfe81SBenoît Canet if (is_max) { 270f17cfe81SBenoît Canet cfg.buckets[index].max = value; 2716f9b6d57SAlberto Garcia /* If max is set, avg should never be 0 */ 2726f9b6d57SAlberto Garcia cfg.buckets[index].avg = MAX(cfg.buckets[index].avg, 1); 273f17cfe81SBenoît Canet } else { 274f17cfe81SBenoît Canet cfg.buckets[index].avg = value; 275f17cfe81SBenoît Canet } 276f17cfe81SBenoît Canet } 277f17cfe81SBenoît Canet 278f17cfe81SBenoît Canet static void test_enabled(void) 279f17cfe81SBenoît Canet { 280f17cfe81SBenoît Canet int i; 281f17cfe81SBenoît Canet 2821588ab5dSAlberto Garcia throttle_config_init(&cfg); 283f17cfe81SBenoît Canet g_assert(!throttle_enabled(&cfg)); 284f17cfe81SBenoît Canet 285f17cfe81SBenoît Canet for (i = 0; i < BUCKETS_COUNT; i++) { 2861588ab5dSAlberto Garcia throttle_config_init(&cfg); 287f17cfe81SBenoît Canet set_cfg_value(false, i, 150); 288f17cfe81SBenoît Canet g_assert(throttle_enabled(&cfg)); 289f17cfe81SBenoît Canet } 290f17cfe81SBenoît Canet 291f17cfe81SBenoît Canet for (i = 0; i < BUCKETS_COUNT; i++) { 2921588ab5dSAlberto Garcia throttle_config_init(&cfg); 293f17cfe81SBenoît Canet set_cfg_value(false, i, -150); 294f17cfe81SBenoît Canet g_assert(!throttle_enabled(&cfg)); 295f17cfe81SBenoît Canet } 296f17cfe81SBenoît Canet } 297f17cfe81SBenoît Canet 298f17cfe81SBenoît Canet /* tests functions for throttle_conflicting */ 299f17cfe81SBenoît Canet 300f17cfe81SBenoît Canet static void test_conflicts_for_one_set(bool is_max, 301f17cfe81SBenoît Canet int total, 302f17cfe81SBenoît Canet int read, 303f17cfe81SBenoît Canet int write) 304f17cfe81SBenoît Canet { 3051588ab5dSAlberto Garcia throttle_config_init(&cfg); 306d5851089SAlberto Garcia g_assert(throttle_is_valid(&cfg, NULL)); 307f17cfe81SBenoît Canet 308f17cfe81SBenoît Canet set_cfg_value(is_max, total, 1); 309f17cfe81SBenoît Canet set_cfg_value(is_max, read, 1); 310d5851089SAlberto Garcia g_assert(!throttle_is_valid(&cfg, NULL)); 311f17cfe81SBenoît Canet 3121588ab5dSAlberto Garcia throttle_config_init(&cfg); 313f17cfe81SBenoît Canet set_cfg_value(is_max, total, 1); 314f17cfe81SBenoît Canet set_cfg_value(is_max, write, 1); 315d5851089SAlberto Garcia g_assert(!throttle_is_valid(&cfg, NULL)); 316f17cfe81SBenoît Canet 3171588ab5dSAlberto Garcia throttle_config_init(&cfg); 318f17cfe81SBenoît Canet set_cfg_value(is_max, total, 1); 319f17cfe81SBenoît Canet set_cfg_value(is_max, read, 1); 320f17cfe81SBenoît Canet set_cfg_value(is_max, write, 1); 321d5851089SAlberto Garcia g_assert(!throttle_is_valid(&cfg, NULL)); 322f17cfe81SBenoît Canet 3231588ab5dSAlberto Garcia throttle_config_init(&cfg); 324f17cfe81SBenoît Canet set_cfg_value(is_max, total, 1); 325d5851089SAlberto Garcia g_assert(throttle_is_valid(&cfg, NULL)); 326f17cfe81SBenoît Canet 3271588ab5dSAlberto Garcia throttle_config_init(&cfg); 328f17cfe81SBenoît Canet set_cfg_value(is_max, read, 1); 329f17cfe81SBenoît Canet set_cfg_value(is_max, write, 1); 330d5851089SAlberto Garcia g_assert(throttle_is_valid(&cfg, NULL)); 331f17cfe81SBenoît Canet } 332f17cfe81SBenoît Canet 333f17cfe81SBenoît Canet static void test_conflicting_config(void) 334f17cfe81SBenoît Canet { 335f17cfe81SBenoît Canet /* bps average conflicts */ 336f17cfe81SBenoît Canet test_conflicts_for_one_set(false, 337f17cfe81SBenoît Canet THROTTLE_BPS_TOTAL, 338f17cfe81SBenoît Canet THROTTLE_BPS_READ, 339f17cfe81SBenoît Canet THROTTLE_BPS_WRITE); 340f17cfe81SBenoît Canet 341f17cfe81SBenoît Canet /* ops average conflicts */ 342f17cfe81SBenoît Canet test_conflicts_for_one_set(false, 343f17cfe81SBenoît Canet THROTTLE_OPS_TOTAL, 344f17cfe81SBenoît Canet THROTTLE_OPS_READ, 345f17cfe81SBenoît Canet THROTTLE_OPS_WRITE); 346f17cfe81SBenoît Canet 347f17cfe81SBenoît Canet /* bps average conflicts */ 348f17cfe81SBenoît Canet test_conflicts_for_one_set(true, 349f17cfe81SBenoît Canet THROTTLE_BPS_TOTAL, 350f17cfe81SBenoît Canet THROTTLE_BPS_READ, 351f17cfe81SBenoît Canet THROTTLE_BPS_WRITE); 352f17cfe81SBenoît Canet /* ops average conflicts */ 353f17cfe81SBenoît Canet test_conflicts_for_one_set(true, 354f17cfe81SBenoît Canet THROTTLE_OPS_TOTAL, 355f17cfe81SBenoît Canet THROTTLE_OPS_READ, 356f17cfe81SBenoît Canet THROTTLE_OPS_WRITE); 357f17cfe81SBenoît Canet } 358f17cfe81SBenoît Canet /* functions to test the throttle_is_valid function */ 359f17cfe81SBenoît Canet static void test_is_valid_for_value(int value, bool should_be_valid) 360f17cfe81SBenoît Canet { 361f17cfe81SBenoît Canet int is_max, index; 362f17cfe81SBenoît Canet for (is_max = 0; is_max < 2; is_max++) { 363f17cfe81SBenoît Canet for (index = 0; index < BUCKETS_COUNT; index++) { 3641588ab5dSAlberto Garcia throttle_config_init(&cfg); 365f17cfe81SBenoît Canet set_cfg_value(is_max, index, value); 36603ba36c8SAlberto Garcia g_assert(throttle_is_valid(&cfg, NULL) == should_be_valid); 367f17cfe81SBenoît Canet } 368f17cfe81SBenoît Canet } 369f17cfe81SBenoît Canet } 370f17cfe81SBenoît Canet 371f17cfe81SBenoît Canet static void test_is_valid(void) 372f17cfe81SBenoît Canet { 373f17cfe81SBenoît Canet /* negative number are invalid */ 374f17cfe81SBenoît Canet test_is_valid_for_value(-1, false); 375f17cfe81SBenoît Canet /* zero are valids */ 376f17cfe81SBenoît Canet test_is_valid_for_value(0, true); 377f17cfe81SBenoît Canet /* positives numers are valids */ 378f17cfe81SBenoît Canet test_is_valid_for_value(1, true); 379f17cfe81SBenoît Canet } 380f17cfe81SBenoît Canet 38192e11a17SStefan Hajnoczi static void test_max_is_missing_limit(void) 38292e11a17SStefan Hajnoczi { 38392e11a17SStefan Hajnoczi int i; 38492e11a17SStefan Hajnoczi 38592e11a17SStefan Hajnoczi for (i = 0; i < BUCKETS_COUNT; i++) { 3861588ab5dSAlberto Garcia throttle_config_init(&cfg); 38792e11a17SStefan Hajnoczi cfg.buckets[i].max = 100; 38892e11a17SStefan Hajnoczi cfg.buckets[i].avg = 0; 389d5851089SAlberto Garcia g_assert(!throttle_is_valid(&cfg, NULL)); 39092e11a17SStefan Hajnoczi 39192e11a17SStefan Hajnoczi cfg.buckets[i].max = 0; 39292e11a17SStefan Hajnoczi cfg.buckets[i].avg = 0; 393d5851089SAlberto Garcia g_assert(throttle_is_valid(&cfg, NULL)); 39492e11a17SStefan Hajnoczi 39592e11a17SStefan Hajnoczi cfg.buckets[i].max = 0; 39692e11a17SStefan Hajnoczi cfg.buckets[i].avg = 100; 397d5851089SAlberto Garcia g_assert(throttle_is_valid(&cfg, NULL)); 39892e11a17SStefan Hajnoczi } 39992e11a17SStefan Hajnoczi } 40092e11a17SStefan Hajnoczi 401f17cfe81SBenoît Canet static void test_have_timer(void) 402f17cfe81SBenoît Canet { 4030e5b0a2dSBenoît Canet /* zero structures */ 404f17cfe81SBenoît Canet memset(&ts, 0, sizeof(ts)); 4050e5b0a2dSBenoît Canet memset(&tt, 0, sizeof(tt)); 406f17cfe81SBenoît Canet 40773f395faSStefan Weil /* no timer set should return false */ 4080e5b0a2dSBenoît Canet g_assert(!throttle_timers_are_initialized(&tt)); 409f17cfe81SBenoît Canet 4100e5b0a2dSBenoît Canet /* init structures */ 4110e5b0a2dSBenoît Canet throttle_init(&ts); 4120e5b0a2dSBenoît Canet throttle_timers_init(&tt, ctx, QEMU_CLOCK_VIRTUAL, 41313af91ebSStefan Hajnoczi read_timer_cb, write_timer_cb, &ts); 414f17cfe81SBenoît Canet 415f17cfe81SBenoît Canet /* timer set by init should return true */ 4160e5b0a2dSBenoît Canet g_assert(throttle_timers_are_initialized(&tt)); 417f17cfe81SBenoît Canet 4180e5b0a2dSBenoît Canet throttle_timers_destroy(&tt); 419f17cfe81SBenoît Canet } 420f17cfe81SBenoît Canet 42122524f72SStefan Hajnoczi static void test_detach_attach(void) 42222524f72SStefan Hajnoczi { 4230e5b0a2dSBenoît Canet /* zero structures */ 42422524f72SStefan Hajnoczi memset(&ts, 0, sizeof(ts)); 4250e5b0a2dSBenoît Canet memset(&tt, 0, sizeof(tt)); 42622524f72SStefan Hajnoczi 42722524f72SStefan Hajnoczi /* init the structure */ 4280e5b0a2dSBenoît Canet throttle_init(&ts); 4290e5b0a2dSBenoît Canet throttle_timers_init(&tt, ctx, QEMU_CLOCK_VIRTUAL, 43022524f72SStefan Hajnoczi read_timer_cb, write_timer_cb, &ts); 43122524f72SStefan Hajnoczi 43222524f72SStefan Hajnoczi /* timer set by init should return true */ 4330e5b0a2dSBenoît Canet g_assert(throttle_timers_are_initialized(&tt)); 43422524f72SStefan Hajnoczi 43522524f72SStefan Hajnoczi /* timer should no longer exist after detaching */ 4360e5b0a2dSBenoît Canet throttle_timers_detach_aio_context(&tt); 4370e5b0a2dSBenoît Canet g_assert(!throttle_timers_are_initialized(&tt)); 43822524f72SStefan Hajnoczi 43922524f72SStefan Hajnoczi /* timer should exist again after attaching */ 4400e5b0a2dSBenoît Canet throttle_timers_attach_aio_context(&tt, ctx); 4410e5b0a2dSBenoît Canet g_assert(throttle_timers_are_initialized(&tt)); 44222524f72SStefan Hajnoczi 4430e5b0a2dSBenoît Canet throttle_timers_destroy(&tt); 44422524f72SStefan Hajnoczi } 44522524f72SStefan Hajnoczi 446f17cfe81SBenoît Canet static bool do_test_accounting(bool is_ops, /* are we testing bps or ops */ 447f17cfe81SBenoît Canet int size, /* size of the operation to do */ 448f17cfe81SBenoît Canet double avg, /* io limit */ 449f17cfe81SBenoît Canet uint64_t op_size, /* ideal size of an io */ 450f17cfe81SBenoît Canet double total_result, 451f17cfe81SBenoît Canet double read_result, 452f17cfe81SBenoît Canet double write_result) 453f17cfe81SBenoît Canet { 454f17cfe81SBenoît Canet BucketType to_test[2][3] = { { THROTTLE_BPS_TOTAL, 455f17cfe81SBenoît Canet THROTTLE_BPS_READ, 456f17cfe81SBenoît Canet THROTTLE_BPS_WRITE, }, 457f17cfe81SBenoît Canet { THROTTLE_OPS_TOTAL, 458f17cfe81SBenoît Canet THROTTLE_OPS_READ, 459f17cfe81SBenoît Canet THROTTLE_OPS_WRITE, } }; 460f17cfe81SBenoît Canet ThrottleConfig cfg; 461f17cfe81SBenoît Canet BucketType index; 462f17cfe81SBenoît Canet int i; 463f17cfe81SBenoît Canet 464f17cfe81SBenoît Canet for (i = 0; i < 3; i++) { 465f17cfe81SBenoît Canet BucketType index = to_test[is_ops][i]; 466f17cfe81SBenoît Canet cfg.buckets[index].avg = avg; 467f17cfe81SBenoît Canet } 468f17cfe81SBenoît Canet 469f17cfe81SBenoît Canet cfg.op_size = op_size; 470f17cfe81SBenoît Canet 4710e5b0a2dSBenoît Canet throttle_init(&ts); 4720e5b0a2dSBenoît Canet throttle_timers_init(&tt, ctx, QEMU_CLOCK_VIRTUAL, 47313af91ebSStefan Hajnoczi read_timer_cb, write_timer_cb, &ts); 4740e5b0a2dSBenoît Canet throttle_config(&ts, &tt, &cfg); 475f17cfe81SBenoît Canet 476f17cfe81SBenoît Canet /* account a read */ 477f17cfe81SBenoît Canet throttle_account(&ts, false, size); 478f17cfe81SBenoît Canet /* account a write */ 479f17cfe81SBenoît Canet throttle_account(&ts, true, size); 480f17cfe81SBenoît Canet 481f17cfe81SBenoît Canet /* check total result */ 482f17cfe81SBenoît Canet index = to_test[is_ops][0]; 483f17cfe81SBenoît Canet if (!double_cmp(ts.cfg.buckets[index].level, total_result)) { 484f17cfe81SBenoît Canet return false; 485f17cfe81SBenoît Canet } 486f17cfe81SBenoît Canet 487f17cfe81SBenoît Canet /* check read result */ 488f17cfe81SBenoît Canet index = to_test[is_ops][1]; 489f17cfe81SBenoît Canet if (!double_cmp(ts.cfg.buckets[index].level, read_result)) { 490f17cfe81SBenoît Canet return false; 491f17cfe81SBenoît Canet } 492f17cfe81SBenoît Canet 493f17cfe81SBenoît Canet /* check write result */ 494f17cfe81SBenoît Canet index = to_test[is_ops][2]; 495f17cfe81SBenoît Canet if (!double_cmp(ts.cfg.buckets[index].level, write_result)) { 496f17cfe81SBenoît Canet return false; 497f17cfe81SBenoît Canet } 498f17cfe81SBenoît Canet 4990e5b0a2dSBenoît Canet throttle_timers_destroy(&tt); 500f17cfe81SBenoît Canet 501f17cfe81SBenoît Canet return true; 502f17cfe81SBenoît Canet } 503f17cfe81SBenoît Canet 504f17cfe81SBenoît Canet static void test_accounting(void) 505f17cfe81SBenoît Canet { 506f17cfe81SBenoît Canet /* tests for bps */ 507f17cfe81SBenoît Canet 508f17cfe81SBenoît Canet /* op of size 1 */ 509f17cfe81SBenoît Canet g_assert(do_test_accounting(false, 510f17cfe81SBenoît Canet 1 * 512, 511f17cfe81SBenoît Canet 150, 512f17cfe81SBenoît Canet 0, 513f17cfe81SBenoît Canet 1024, 514f17cfe81SBenoît Canet 512, 515f17cfe81SBenoît Canet 512)); 516f17cfe81SBenoît Canet 517f17cfe81SBenoît Canet /* op of size 2 */ 518f17cfe81SBenoît Canet g_assert(do_test_accounting(false, 519f17cfe81SBenoît Canet 2 * 512, 520f17cfe81SBenoît Canet 150, 521f17cfe81SBenoît Canet 0, 522f17cfe81SBenoît Canet 2048, 523f17cfe81SBenoît Canet 1024, 524f17cfe81SBenoît Canet 1024)); 525f17cfe81SBenoît Canet 526f17cfe81SBenoît Canet /* op of size 2 and orthogonal parameter change */ 527f17cfe81SBenoît Canet g_assert(do_test_accounting(false, 528f17cfe81SBenoît Canet 2 * 512, 529f17cfe81SBenoît Canet 150, 530f17cfe81SBenoît Canet 17, 531f17cfe81SBenoît Canet 2048, 532f17cfe81SBenoît Canet 1024, 533f17cfe81SBenoît Canet 1024)); 534f17cfe81SBenoît Canet 535f17cfe81SBenoît Canet 536f17cfe81SBenoît Canet /* tests for ops */ 537f17cfe81SBenoît Canet 538f17cfe81SBenoît Canet /* op of size 1 */ 539f17cfe81SBenoît Canet g_assert(do_test_accounting(true, 540f17cfe81SBenoît Canet 1 * 512, 541f17cfe81SBenoît Canet 150, 542f17cfe81SBenoît Canet 0, 543f17cfe81SBenoît Canet 2, 544f17cfe81SBenoît Canet 1, 545f17cfe81SBenoît Canet 1)); 546f17cfe81SBenoît Canet 547f17cfe81SBenoît Canet /* op of size 2 */ 548f17cfe81SBenoît Canet g_assert(do_test_accounting(true, 549f17cfe81SBenoît Canet 2 * 512, 550f17cfe81SBenoît Canet 150, 551f17cfe81SBenoît Canet 0, 552f17cfe81SBenoît Canet 2, 553f17cfe81SBenoît Canet 1, 554f17cfe81SBenoît Canet 1)); 555f17cfe81SBenoît Canet 556f17cfe81SBenoît Canet /* jumbo op accounting fragmentation : size 64 with op size of 13 units */ 557f17cfe81SBenoît Canet g_assert(do_test_accounting(true, 558f17cfe81SBenoît Canet 64 * 512, 559f17cfe81SBenoît Canet 150, 560f17cfe81SBenoît Canet 13 * 512, 561f17cfe81SBenoît Canet (64.0 * 2) / 13, 562f17cfe81SBenoît Canet (64.0 / 13), 563f17cfe81SBenoît Canet (64.0 / 13))); 564f17cfe81SBenoît Canet 565f17cfe81SBenoît Canet /* same with orthogonal parameters changes */ 566f17cfe81SBenoît Canet g_assert(do_test_accounting(true, 567f17cfe81SBenoît Canet 64 * 512, 568f17cfe81SBenoît Canet 300, 569f17cfe81SBenoît Canet 13 * 512, 570f17cfe81SBenoît Canet (64.0 * 2) / 13, 571f17cfe81SBenoît Canet (64.0 / 13), 572f17cfe81SBenoît Canet (64.0 / 13))); 573f17cfe81SBenoît Canet } 574f17cfe81SBenoît Canet 5751fee955fSAlberto Garcia static void test_groups(void) 5761fee955fSAlberto Garcia { 5771fee955fSAlberto Garcia ThrottleConfig cfg1, cfg2; 578a5614993SKevin Wolf BlockBackend *blk1, *blk2, *blk3; 5791fee955fSAlberto Garcia BlockDriverState *bdrv1, *bdrv2, *bdrv3; 5801fee955fSAlberto Garcia 581a5614993SKevin Wolf blk1 = blk_new_with_bs(&error_abort); 582a5614993SKevin Wolf blk2 = blk_new_with_bs(&error_abort); 583a5614993SKevin Wolf blk3 = blk_new_with_bs(&error_abort); 584a5614993SKevin Wolf 585a5614993SKevin Wolf bdrv1 = blk_bs(blk1); 586a5614993SKevin Wolf bdrv2 = blk_bs(blk2); 587a5614993SKevin Wolf bdrv3 = blk_bs(blk3); 5881fee955fSAlberto Garcia 5891fee955fSAlberto Garcia g_assert(bdrv1->throttle_state == NULL); 5901fee955fSAlberto Garcia g_assert(bdrv2->throttle_state == NULL); 5911fee955fSAlberto Garcia g_assert(bdrv3->throttle_state == NULL); 5921fee955fSAlberto Garcia 593*31dce3ccSKevin Wolf throttle_group_register_blk(blk1, "bar"); 594*31dce3ccSKevin Wolf throttle_group_register_blk(blk2, "foo"); 595*31dce3ccSKevin Wolf throttle_group_register_blk(blk3, "bar"); 5961fee955fSAlberto Garcia 5971fee955fSAlberto Garcia g_assert(bdrv1->throttle_state != NULL); 5981fee955fSAlberto Garcia g_assert(bdrv2->throttle_state != NULL); 5991fee955fSAlberto Garcia g_assert(bdrv3->throttle_state != NULL); 6001fee955fSAlberto Garcia 6011fee955fSAlberto Garcia g_assert(!strcmp(throttle_group_get_name(bdrv1), "bar")); 6021fee955fSAlberto Garcia g_assert(!strcmp(throttle_group_get_name(bdrv2), "foo")); 6031fee955fSAlberto Garcia g_assert(bdrv1->throttle_state == bdrv3->throttle_state); 6041fee955fSAlberto Garcia 6051fee955fSAlberto Garcia /* Setting the config of a group member affects the whole group */ 6061588ab5dSAlberto Garcia throttle_config_init(&cfg1); 6071fee955fSAlberto Garcia cfg1.buckets[THROTTLE_BPS_READ].avg = 500000; 6081fee955fSAlberto Garcia cfg1.buckets[THROTTLE_BPS_WRITE].avg = 285000; 6091fee955fSAlberto Garcia cfg1.buckets[THROTTLE_OPS_READ].avg = 20000; 6101fee955fSAlberto Garcia cfg1.buckets[THROTTLE_OPS_WRITE].avg = 12000; 6111fee955fSAlberto Garcia throttle_group_config(bdrv1, &cfg1); 6121fee955fSAlberto Garcia 6131fee955fSAlberto Garcia throttle_group_get_config(bdrv1, &cfg1); 6141fee955fSAlberto Garcia throttle_group_get_config(bdrv3, &cfg2); 6151fee955fSAlberto Garcia g_assert(!memcmp(&cfg1, &cfg2, sizeof(cfg1))); 6161fee955fSAlberto Garcia 6171fee955fSAlberto Garcia cfg2.buckets[THROTTLE_BPS_READ].avg = 4547; 6181fee955fSAlberto Garcia cfg2.buckets[THROTTLE_BPS_WRITE].avg = 1349; 6191fee955fSAlberto Garcia cfg2.buckets[THROTTLE_OPS_READ].avg = 123; 6201fee955fSAlberto Garcia cfg2.buckets[THROTTLE_OPS_WRITE].avg = 86; 6211fee955fSAlberto Garcia throttle_group_config(bdrv3, &cfg1); 6221fee955fSAlberto Garcia 6231fee955fSAlberto Garcia throttle_group_get_config(bdrv1, &cfg1); 6241fee955fSAlberto Garcia throttle_group_get_config(bdrv3, &cfg2); 6251fee955fSAlberto Garcia g_assert(!memcmp(&cfg1, &cfg2, sizeof(cfg1))); 6261fee955fSAlberto Garcia 627*31dce3ccSKevin Wolf throttle_group_unregister_blk(blk1); 628*31dce3ccSKevin Wolf throttle_group_unregister_blk(blk2); 629*31dce3ccSKevin Wolf throttle_group_unregister_blk(blk3); 6301fee955fSAlberto Garcia 6311fee955fSAlberto Garcia g_assert(bdrv1->throttle_state == NULL); 6321fee955fSAlberto Garcia g_assert(bdrv2->throttle_state == NULL); 6331fee955fSAlberto Garcia g_assert(bdrv3->throttle_state == NULL); 6341fee955fSAlberto Garcia } 6351fee955fSAlberto Garcia 636f17cfe81SBenoît Canet int main(int argc, char **argv) 637f17cfe81SBenoît Canet { 63873eaa047SMarkus Armbruster qemu_init_main_loop(&error_fatal); 6391fee955fSAlberto Garcia ctx = qemu_get_aio_context(); 6401fee955fSAlberto Garcia bdrv_init(); 64113af91ebSStefan Hajnoczi 642f17cfe81SBenoît Canet do {} while (g_main_context_iteration(NULL, false)); 643f17cfe81SBenoît Canet 644f17cfe81SBenoît Canet /* tests in the same order as the header function declarations */ 645f17cfe81SBenoît Canet g_test_init(&argc, &argv, NULL); 646f17cfe81SBenoît Canet g_test_add_func("/throttle/leak_bucket", test_leak_bucket); 647f17cfe81SBenoît Canet g_test_add_func("/throttle/compute_wait", test_compute_wait); 648f17cfe81SBenoît Canet g_test_add_func("/throttle/init", test_init); 649f17cfe81SBenoît Canet g_test_add_func("/throttle/destroy", test_destroy); 650f17cfe81SBenoît Canet g_test_add_func("/throttle/have_timer", test_have_timer); 65122524f72SStefan Hajnoczi g_test_add_func("/throttle/detach_attach", test_detach_attach); 652f17cfe81SBenoît Canet g_test_add_func("/throttle/config/enabled", test_enabled); 653f17cfe81SBenoît Canet g_test_add_func("/throttle/config/conflicting", test_conflicting_config); 654f17cfe81SBenoît Canet g_test_add_func("/throttle/config/is_valid", test_is_valid); 65592e11a17SStefan Hajnoczi g_test_add_func("/throttle/config/max", test_max_is_missing_limit); 656f17cfe81SBenoît Canet g_test_add_func("/throttle/config_functions", test_config_functions); 657f17cfe81SBenoît Canet g_test_add_func("/throttle/accounting", test_accounting); 6581fee955fSAlberto Garcia g_test_add_func("/throttle/groups", test_groups); 659f17cfe81SBenoît Canet return g_test_run(); 660f17cfe81SBenoît Canet } 661f17cfe81SBenoît Canet 662