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" 2331dce3ccSKevin 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 401*8860eabdSStefan Hajnoczi static void test_iops_size_is_missing_limit(void) 402*8860eabdSStefan Hajnoczi { 403*8860eabdSStefan Hajnoczi /* A total/read/write iops limit is required */ 404*8860eabdSStefan Hajnoczi throttle_config_init(&cfg); 405*8860eabdSStefan Hajnoczi cfg.op_size = 4096; 406*8860eabdSStefan Hajnoczi g_assert(!throttle_is_valid(&cfg, NULL)); 407*8860eabdSStefan Hajnoczi } 408*8860eabdSStefan Hajnoczi 409f17cfe81SBenoît Canet static void test_have_timer(void) 410f17cfe81SBenoît Canet { 4110e5b0a2dSBenoît Canet /* zero structures */ 412f17cfe81SBenoît Canet memset(&ts, 0, sizeof(ts)); 4130e5b0a2dSBenoît Canet memset(&tt, 0, sizeof(tt)); 414f17cfe81SBenoît Canet 41573f395faSStefan Weil /* no timer set should return false */ 4160e5b0a2dSBenoît Canet g_assert(!throttle_timers_are_initialized(&tt)); 417f17cfe81SBenoît Canet 4180e5b0a2dSBenoît Canet /* init structures */ 4190e5b0a2dSBenoît Canet throttle_init(&ts); 4200e5b0a2dSBenoît Canet throttle_timers_init(&tt, ctx, QEMU_CLOCK_VIRTUAL, 42113af91ebSStefan Hajnoczi read_timer_cb, write_timer_cb, &ts); 422f17cfe81SBenoît Canet 423f17cfe81SBenoît Canet /* timer set by init should return true */ 4240e5b0a2dSBenoît Canet g_assert(throttle_timers_are_initialized(&tt)); 425f17cfe81SBenoît Canet 4260e5b0a2dSBenoît Canet throttle_timers_destroy(&tt); 427f17cfe81SBenoît Canet } 428f17cfe81SBenoît Canet 42922524f72SStefan Hajnoczi static void test_detach_attach(void) 43022524f72SStefan Hajnoczi { 4310e5b0a2dSBenoît Canet /* zero structures */ 43222524f72SStefan Hajnoczi memset(&ts, 0, sizeof(ts)); 4330e5b0a2dSBenoît Canet memset(&tt, 0, sizeof(tt)); 43422524f72SStefan Hajnoczi 43522524f72SStefan Hajnoczi /* init the structure */ 4360e5b0a2dSBenoît Canet throttle_init(&ts); 4370e5b0a2dSBenoît Canet throttle_timers_init(&tt, ctx, QEMU_CLOCK_VIRTUAL, 43822524f72SStefan Hajnoczi read_timer_cb, write_timer_cb, &ts); 43922524f72SStefan Hajnoczi 44022524f72SStefan Hajnoczi /* timer set by init should return true */ 4410e5b0a2dSBenoît Canet g_assert(throttle_timers_are_initialized(&tt)); 44222524f72SStefan Hajnoczi 44322524f72SStefan Hajnoczi /* timer should no longer exist after detaching */ 4440e5b0a2dSBenoît Canet throttle_timers_detach_aio_context(&tt); 4450e5b0a2dSBenoît Canet g_assert(!throttle_timers_are_initialized(&tt)); 44622524f72SStefan Hajnoczi 44722524f72SStefan Hajnoczi /* timer should exist again after attaching */ 4480e5b0a2dSBenoît Canet throttle_timers_attach_aio_context(&tt, ctx); 4490e5b0a2dSBenoît Canet g_assert(throttle_timers_are_initialized(&tt)); 45022524f72SStefan Hajnoczi 4510e5b0a2dSBenoît Canet throttle_timers_destroy(&tt); 45222524f72SStefan Hajnoczi } 45322524f72SStefan Hajnoczi 454f17cfe81SBenoît Canet static bool do_test_accounting(bool is_ops, /* are we testing bps or ops */ 455f17cfe81SBenoît Canet int size, /* size of the operation to do */ 456f17cfe81SBenoît Canet double avg, /* io limit */ 457f17cfe81SBenoît Canet uint64_t op_size, /* ideal size of an io */ 458f17cfe81SBenoît Canet double total_result, 459f17cfe81SBenoît Canet double read_result, 460f17cfe81SBenoît Canet double write_result) 461f17cfe81SBenoît Canet { 462f17cfe81SBenoît Canet BucketType to_test[2][3] = { { THROTTLE_BPS_TOTAL, 463f17cfe81SBenoît Canet THROTTLE_BPS_READ, 464f17cfe81SBenoît Canet THROTTLE_BPS_WRITE, }, 465f17cfe81SBenoît Canet { THROTTLE_OPS_TOTAL, 466f17cfe81SBenoît Canet THROTTLE_OPS_READ, 467f17cfe81SBenoît Canet THROTTLE_OPS_WRITE, } }; 468f17cfe81SBenoît Canet ThrottleConfig cfg; 469f17cfe81SBenoît Canet BucketType index; 470f17cfe81SBenoît Canet int i; 471f17cfe81SBenoît Canet 472f17cfe81SBenoît Canet for (i = 0; i < 3; i++) { 473f17cfe81SBenoît Canet BucketType index = to_test[is_ops][i]; 474f17cfe81SBenoît Canet cfg.buckets[index].avg = avg; 475f17cfe81SBenoît Canet } 476f17cfe81SBenoît Canet 477f17cfe81SBenoît Canet cfg.op_size = op_size; 478f17cfe81SBenoît Canet 4790e5b0a2dSBenoît Canet throttle_init(&ts); 4800e5b0a2dSBenoît Canet throttle_timers_init(&tt, ctx, QEMU_CLOCK_VIRTUAL, 48113af91ebSStefan Hajnoczi read_timer_cb, write_timer_cb, &ts); 4820e5b0a2dSBenoît Canet throttle_config(&ts, &tt, &cfg); 483f17cfe81SBenoît Canet 484f17cfe81SBenoît Canet /* account a read */ 485f17cfe81SBenoît Canet throttle_account(&ts, false, size); 486f17cfe81SBenoît Canet /* account a write */ 487f17cfe81SBenoît Canet throttle_account(&ts, true, size); 488f17cfe81SBenoît Canet 489f17cfe81SBenoît Canet /* check total result */ 490f17cfe81SBenoît Canet index = to_test[is_ops][0]; 491f17cfe81SBenoît Canet if (!double_cmp(ts.cfg.buckets[index].level, total_result)) { 492f17cfe81SBenoît Canet return false; 493f17cfe81SBenoît Canet } 494f17cfe81SBenoît Canet 495f17cfe81SBenoît Canet /* check read result */ 496f17cfe81SBenoît Canet index = to_test[is_ops][1]; 497f17cfe81SBenoît Canet if (!double_cmp(ts.cfg.buckets[index].level, read_result)) { 498f17cfe81SBenoît Canet return false; 499f17cfe81SBenoît Canet } 500f17cfe81SBenoît Canet 501f17cfe81SBenoît Canet /* check write result */ 502f17cfe81SBenoît Canet index = to_test[is_ops][2]; 503f17cfe81SBenoît Canet if (!double_cmp(ts.cfg.buckets[index].level, write_result)) { 504f17cfe81SBenoît Canet return false; 505f17cfe81SBenoît Canet } 506f17cfe81SBenoît Canet 5070e5b0a2dSBenoît Canet throttle_timers_destroy(&tt); 508f17cfe81SBenoît Canet 509f17cfe81SBenoît Canet return true; 510f17cfe81SBenoît Canet } 511f17cfe81SBenoît Canet 512f17cfe81SBenoît Canet static void test_accounting(void) 513f17cfe81SBenoît Canet { 514f17cfe81SBenoît Canet /* tests for bps */ 515f17cfe81SBenoît Canet 516f17cfe81SBenoît Canet /* op of size 1 */ 517f17cfe81SBenoît Canet g_assert(do_test_accounting(false, 518f17cfe81SBenoît Canet 1 * 512, 519f17cfe81SBenoît Canet 150, 520f17cfe81SBenoît Canet 0, 521f17cfe81SBenoît Canet 1024, 522f17cfe81SBenoît Canet 512, 523f17cfe81SBenoît Canet 512)); 524f17cfe81SBenoît Canet 525f17cfe81SBenoît Canet /* op of size 2 */ 526f17cfe81SBenoît Canet g_assert(do_test_accounting(false, 527f17cfe81SBenoît Canet 2 * 512, 528f17cfe81SBenoît Canet 150, 529f17cfe81SBenoît Canet 0, 530f17cfe81SBenoît Canet 2048, 531f17cfe81SBenoît Canet 1024, 532f17cfe81SBenoît Canet 1024)); 533f17cfe81SBenoît Canet 534f17cfe81SBenoît Canet /* op of size 2 and orthogonal parameter change */ 535f17cfe81SBenoît Canet g_assert(do_test_accounting(false, 536f17cfe81SBenoît Canet 2 * 512, 537f17cfe81SBenoît Canet 150, 538f17cfe81SBenoît Canet 17, 539f17cfe81SBenoît Canet 2048, 540f17cfe81SBenoît Canet 1024, 541f17cfe81SBenoît Canet 1024)); 542f17cfe81SBenoît Canet 543f17cfe81SBenoît Canet 544f17cfe81SBenoît Canet /* tests for ops */ 545f17cfe81SBenoît Canet 546f17cfe81SBenoît Canet /* op of size 1 */ 547f17cfe81SBenoît Canet g_assert(do_test_accounting(true, 548f17cfe81SBenoît Canet 1 * 512, 549f17cfe81SBenoît Canet 150, 550f17cfe81SBenoît Canet 0, 551f17cfe81SBenoît Canet 2, 552f17cfe81SBenoît Canet 1, 553f17cfe81SBenoît Canet 1)); 554f17cfe81SBenoît Canet 555f17cfe81SBenoît Canet /* op of size 2 */ 556f17cfe81SBenoît Canet g_assert(do_test_accounting(true, 557f17cfe81SBenoît Canet 2 * 512, 558f17cfe81SBenoît Canet 150, 559f17cfe81SBenoît Canet 0, 560f17cfe81SBenoît Canet 2, 561f17cfe81SBenoît Canet 1, 562f17cfe81SBenoît Canet 1)); 563f17cfe81SBenoît Canet 564f17cfe81SBenoît Canet /* jumbo op accounting fragmentation : size 64 with op size of 13 units */ 565f17cfe81SBenoît Canet g_assert(do_test_accounting(true, 566f17cfe81SBenoît Canet 64 * 512, 567f17cfe81SBenoît Canet 150, 568f17cfe81SBenoît Canet 13 * 512, 569f17cfe81SBenoît Canet (64.0 * 2) / 13, 570f17cfe81SBenoît Canet (64.0 / 13), 571f17cfe81SBenoît Canet (64.0 / 13))); 572f17cfe81SBenoît Canet 573f17cfe81SBenoît Canet /* same with orthogonal parameters changes */ 574f17cfe81SBenoît Canet g_assert(do_test_accounting(true, 575f17cfe81SBenoît Canet 64 * 512, 576f17cfe81SBenoît Canet 300, 577f17cfe81SBenoît Canet 13 * 512, 578f17cfe81SBenoît Canet (64.0 * 2) / 13, 579f17cfe81SBenoît Canet (64.0 / 13), 580f17cfe81SBenoît Canet (64.0 / 13))); 581f17cfe81SBenoît Canet } 582f17cfe81SBenoît Canet 5831fee955fSAlberto Garcia static void test_groups(void) 5841fee955fSAlberto Garcia { 5851fee955fSAlberto Garcia ThrottleConfig cfg1, cfg2; 586a5614993SKevin Wolf BlockBackend *blk1, *blk2, *blk3; 58727ccdd52SKevin Wolf BlockBackendPublic *blkp1, *blkp2, *blkp3; 5881fee955fSAlberto Garcia 589109525adSMax Reitz blk1 = blk_new(); 590109525adSMax Reitz blk2 = blk_new(); 591109525adSMax Reitz blk3 = blk_new(); 592a5614993SKevin Wolf 59327ccdd52SKevin Wolf blkp1 = blk_get_public(blk1); 59427ccdd52SKevin Wolf blkp2 = blk_get_public(blk2); 59527ccdd52SKevin Wolf blkp3 = blk_get_public(blk3); 59627ccdd52SKevin Wolf 59727ccdd52SKevin Wolf g_assert(blkp1->throttle_state == NULL); 59827ccdd52SKevin Wolf g_assert(blkp2->throttle_state == NULL); 59927ccdd52SKevin Wolf g_assert(blkp3->throttle_state == NULL); 6001fee955fSAlberto Garcia 60131dce3ccSKevin Wolf throttle_group_register_blk(blk1, "bar"); 60231dce3ccSKevin Wolf throttle_group_register_blk(blk2, "foo"); 60331dce3ccSKevin Wolf throttle_group_register_blk(blk3, "bar"); 6041fee955fSAlberto Garcia 60527ccdd52SKevin Wolf g_assert(blkp1->throttle_state != NULL); 60627ccdd52SKevin Wolf g_assert(blkp2->throttle_state != NULL); 60727ccdd52SKevin Wolf g_assert(blkp3->throttle_state != NULL); 6081fee955fSAlberto Garcia 60949d2165dSKevin Wolf g_assert(!strcmp(throttle_group_get_name(blk1), "bar")); 61049d2165dSKevin Wolf g_assert(!strcmp(throttle_group_get_name(blk2), "foo")); 61127ccdd52SKevin Wolf g_assert(blkp1->throttle_state == blkp3->throttle_state); 6121fee955fSAlberto Garcia 6131fee955fSAlberto Garcia /* Setting the config of a group member affects the whole group */ 6141588ab5dSAlberto Garcia throttle_config_init(&cfg1); 6151fee955fSAlberto Garcia cfg1.buckets[THROTTLE_BPS_READ].avg = 500000; 6161fee955fSAlberto Garcia cfg1.buckets[THROTTLE_BPS_WRITE].avg = 285000; 6171fee955fSAlberto Garcia cfg1.buckets[THROTTLE_OPS_READ].avg = 20000; 6181fee955fSAlberto Garcia cfg1.buckets[THROTTLE_OPS_WRITE].avg = 12000; 61997148076SKevin Wolf throttle_group_config(blk1, &cfg1); 6201fee955fSAlberto Garcia 62197148076SKevin Wolf throttle_group_get_config(blk1, &cfg1); 62297148076SKevin Wolf throttle_group_get_config(blk3, &cfg2); 6231fee955fSAlberto Garcia g_assert(!memcmp(&cfg1, &cfg2, sizeof(cfg1))); 6241fee955fSAlberto Garcia 6251fee955fSAlberto Garcia cfg2.buckets[THROTTLE_BPS_READ].avg = 4547; 6261fee955fSAlberto Garcia cfg2.buckets[THROTTLE_BPS_WRITE].avg = 1349; 6271fee955fSAlberto Garcia cfg2.buckets[THROTTLE_OPS_READ].avg = 123; 6281fee955fSAlberto Garcia cfg2.buckets[THROTTLE_OPS_WRITE].avg = 86; 62997148076SKevin Wolf throttle_group_config(blk3, &cfg1); 6301fee955fSAlberto Garcia 63197148076SKevin Wolf throttle_group_get_config(blk1, &cfg1); 63297148076SKevin Wolf throttle_group_get_config(blk3, &cfg2); 6331fee955fSAlberto Garcia g_assert(!memcmp(&cfg1, &cfg2, sizeof(cfg1))); 6341fee955fSAlberto Garcia 63531dce3ccSKevin Wolf throttle_group_unregister_blk(blk1); 63631dce3ccSKevin Wolf throttle_group_unregister_blk(blk2); 63731dce3ccSKevin Wolf throttle_group_unregister_blk(blk3); 6381fee955fSAlberto Garcia 63927ccdd52SKevin Wolf g_assert(blkp1->throttle_state == NULL); 64027ccdd52SKevin Wolf g_assert(blkp2->throttle_state == NULL); 64127ccdd52SKevin Wolf g_assert(blkp3->throttle_state == NULL); 6421fee955fSAlberto Garcia } 6431fee955fSAlberto Garcia 644f17cfe81SBenoît Canet int main(int argc, char **argv) 645f17cfe81SBenoît Canet { 64673eaa047SMarkus Armbruster qemu_init_main_loop(&error_fatal); 6471fee955fSAlberto Garcia ctx = qemu_get_aio_context(); 6481fee955fSAlberto Garcia bdrv_init(); 64913af91ebSStefan Hajnoczi 650f17cfe81SBenoît Canet do {} while (g_main_context_iteration(NULL, false)); 651f17cfe81SBenoît Canet 652f17cfe81SBenoît Canet /* tests in the same order as the header function declarations */ 653f17cfe81SBenoît Canet g_test_init(&argc, &argv, NULL); 654f17cfe81SBenoît Canet g_test_add_func("/throttle/leak_bucket", test_leak_bucket); 655f17cfe81SBenoît Canet g_test_add_func("/throttle/compute_wait", test_compute_wait); 656f17cfe81SBenoît Canet g_test_add_func("/throttle/init", test_init); 657f17cfe81SBenoît Canet g_test_add_func("/throttle/destroy", test_destroy); 658f17cfe81SBenoît Canet g_test_add_func("/throttle/have_timer", test_have_timer); 65922524f72SStefan Hajnoczi g_test_add_func("/throttle/detach_attach", test_detach_attach); 660f17cfe81SBenoît Canet g_test_add_func("/throttle/config/enabled", test_enabled); 661f17cfe81SBenoît Canet g_test_add_func("/throttle/config/conflicting", test_conflicting_config); 662f17cfe81SBenoît Canet g_test_add_func("/throttle/config/is_valid", test_is_valid); 66392e11a17SStefan Hajnoczi g_test_add_func("/throttle/config/max", test_max_is_missing_limit); 664*8860eabdSStefan Hajnoczi g_test_add_func("/throttle/config/iops_size", 665*8860eabdSStefan Hajnoczi test_iops_size_is_missing_limit); 666f17cfe81SBenoît Canet g_test_add_func("/throttle/config_functions", test_config_functions); 667f17cfe81SBenoît Canet g_test_add_func("/throttle/accounting", test_accounting); 6681fee955fSAlberto Garcia g_test_add_func("/throttle/groups", test_groups); 669f17cfe81SBenoît Canet return g_test_run(); 670f17cfe81SBenoît Canet } 671f17cfe81SBenoît Canet 672