/*
* Copyright (c) 2019 Vojtech Horky
* 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.
* - The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
*/
/** @addtogroup hbench
* @{
*/
#include <fibril_synch.h>
#include <stdatomic.h>
#include "../hbench.h"
/*
* Simple benchmark for fibril mutexes. There are two fibrils that compete
* over the same mutex as that is the simplest scenario.
*/
typedef struct {
fibril_mutex_t mutex;
uint64_t counter;
atomic_bool done;
} shared_t;
static errno_t competitor(void *arg)
{
shared_t *shared = arg;
fibril_detach(fibril_get_id());
while (true) {
fibril_mutex_lock(&shared->mutex);
uint64_t local = shared->counter;
fibril_mutex_unlock(&shared->mutex);
if (local == 0) {
break;
}
}
atomic_store(&shared->done, true);
return EOK;
}
static bool runner(bench_env_t *env, bench_run_t *run, uint64_t size)
{
shared_t shared;
fibril_mutex_initialize(&shared.mutex);
shared.counter = size;
atomic_store(&shared.done, false);
fid_t other = fibril_create(competitor, &shared);
fibril_add_ready(other);
bench_run_start(run);
for (uint64_t i = 0; i < size; i++) {
fibril_mutex_lock(&shared.mutex);
shared.counter--;
fibril_mutex_unlock(&shared.mutex);
}
bench_run_stop(run);
while (!atomic_load(&shared.done)) {
fibril_yield();
}
return true;
}
benchmark_t benchmark_fibril_mutex = {
.name = "fibril_mutex",
.desc = "Speed of mutex lock/unlock operations",
.entry = &runner,
.setup = NULL,
.teardown = NULL
};
/** @}
*/