Skip to content

Commit

Permalink
Fix data race and format the code.
Browse files Browse the repository at this point in the history
  • Loading branch information
efenniht committed Apr 18, 2020
1 parent 927c334 commit 1203c38
Show file tree
Hide file tree
Showing 8 changed files with 56 additions and 46 deletions.
1 change: 1 addition & 0 deletions inc/hf/vcpu.h
Original file line number Diff line number Diff line change
Expand Up @@ -93,6 +93,7 @@ struct vcpu_execution_locked {

struct vcpu_execution_locked vcpu_lock(struct vcpu *vcpu);
void vcpu_unlock(struct vcpu_execution_locked *locked);
bool vcpu_try_lock(struct vcpu *vcpu, struct vcpu_execution_locked *locked);
void vcpu_init(struct vcpu *vcpu, struct vm *vm);
void vcpu_on(struct vcpu_execution_locked vcpu, ipaddr_t entry, uintreg_t arg);
spci_vcpu_index_t vcpu_index(const struct vcpu *vcpu);
Expand Down
12 changes: 7 additions & 5 deletions src/api.c
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,8 @@
* acquisition of locks held concurrently by the same physical CPU. Our current
* ordering requirements are as follows:
*
* vcpu::execution_lock -> vm::lock -> vcpu::interrupts_lock -> mm_stage1_lock -> dlog sl
* vcpu::execution_lock -> vm::lock -> vcpu::interrupts_lock -> mm_stage1_lock
* -> dlog sl
*
* Locks of the same kind require the lock of lowest address to be locked first,
* see `sl_lock_both()`.
Expand Down Expand Up @@ -275,7 +276,9 @@ spci_vcpu_count_t api_vcpu_get_count(spci_vm_id_t vm_id,
*/
void api_regs_state_saved(struct vcpu *vcpu)
{
sl_unlock(&vcpu->execution_lock);
if (vcpu->vm->id != HF_PRIMARY_VM_ID) {
sl_unlock(&vcpu->execution_lock);
}
}

/**
Expand Down Expand Up @@ -418,7 +421,7 @@ static bool api_vcpu_prepare_run(const struct vcpu *current, struct vcpu *vcpu,
* until this has finished, so count this state as still running for the
* purposes of this check.
*/
if (sl_try_lock(&vcpu->execution_lock)) {
if (!sl_try_lock(&vcpu->execution_lock)) {
/*
* vCPU is running on another pCPU.
*
Expand All @@ -427,8 +430,7 @@ static bool api_vcpu_prepare_run(const struct vcpu *current, struct vcpu *vcpu,
* return the sleep duration if needed.
*/
*run_ret = spci_error(SPCI_BUSY);
ret = false;
goto out;
return false;
}

if (atomic_load_explicit(&vcpu->vm->aborting, memory_order_relaxed)) {
Expand Down
8 changes: 0 additions & 8 deletions src/arch/aarch64/hypervisor/handler.c
Original file line number Diff line number Diff line change
Expand Up @@ -52,14 +52,6 @@
*/
#define CLIENT_ID_MASK UINT64_C(0xffff)

/**
* Returns a reference to the currently executing vCPU.
*/
static struct vcpu *current(void)
{
return (struct vcpu *)read_msr(tpidr_el2);
}

/**
* Saves the state of per-vCPU peripherals, such as the virtual timer, and
* informs the arch-independent sections that registers have been saved.
Expand Down
21 changes: 16 additions & 5 deletions src/arch/aarch64/hypervisor/psci_handler.c
Original file line number Diff line number Diff line change
Expand Up @@ -296,7 +296,8 @@ bool psci_secondary_vm_handler(struct vcpu *vcpu, uint32_t func, uintreg_t arg0,
cpu_id_t target_affinity = arg0;
uint32_t lowest_affinity_level = arg1;
struct vm *vm = vcpu->vm;
struct vcpu_execution_locked target_vcpu;
struct vcpu *target_vcpu;
struct vcpu_execution_locked target_vcpu_locked;
spci_vcpu_index_t target_vcpu_index =
vcpu_id_to_index(target_affinity);

Expand All @@ -311,10 +312,20 @@ bool psci_secondary_vm_handler(struct vcpu *vcpu, uint32_t func, uintreg_t arg0,
break;
}

target_vcpu = vcpu_lock(vm_get_vcpu(vm, target_vcpu_index));
*ret = vcpu_is_off(target_vcpu) ? PSCI_RETURN_OFF
: PSCI_RETURN_ON;
vcpu_unlock(&target_vcpu);
target_vcpu = vm_get_vcpu(vm, target_vcpu_index);
if (target_vcpu == current()) {
*ret = PSCI_RETURN_ON;
break;
}

if (!vcpu_try_lock(target_vcpu, &target_vcpu_locked)) {
*ret = PSCI_RETURN_ON;
break;
}

*ret = vcpu_is_off(target_vcpu_locked) ? PSCI_RETURN_OFF
: PSCI_RETURN_ON;
vcpu_unlock(&target_vcpu_locked);
break;
}

Expand Down
10 changes: 10 additions & 0 deletions src/arch/aarch64/hypervisor/psci_handler.h
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,16 @@

#include "hf/cpu.h"

#include "msr.h"

bool psci_handler(struct vcpu *vcpu, uint32_t func, uintreg_t arg0,
uintreg_t arg1, uintreg_t arg2, uintreg_t *ret,
struct vcpu **next);

/**
* Returns a reference to the currently executing vCPU.
*/
static inline struct vcpu *current(void)
{
return (struct vcpu *)read_msr(tpidr_el2);
}
34 changes: 7 additions & 27 deletions src/arch/aarch64/inc/hf/arch/spinlock.h
Original file line number Diff line number Diff line change
Expand Up @@ -28,50 +28,30 @@
* the guarantees provided by atomic instructions introduced in Armv8.1 LSE.
*/

#include <stdint.h>
#include <stdatomic.h>
#include <stdint.h>

#include "hf/arch/types.h"

struct spinlock {
volatile uint32_t v;
atomic_flag v;
};

#define SPINLOCK_INIT ((struct spinlock){.v = 0})

static inline void sl_lock(struct spinlock *l)
{
register uintreg_t tmp1;
register uintreg_t tmp2;

/*
* Acquire the lock with a LDAXR/STXR pair (acquire semantics on the
* load instruction). Pause using WFE if the lock is currently taken.
* This is NOT guaranteed to make progress.
*/
__asm__ volatile(
" mov %w2, #1\n"
" sevl\n" /* set event bit */
"1: wfe\n" /* wait for event, clear event bit */
"2: ldaxr %w1, [%3]\n" /* load lock value */
" cbnz %w1, 1b\n" /* if lock taken, goto WFE */
" stxr %w1, %w2, [%3]\n" /* try to take lock */
" cbnz %w1, 2b\n" /* loop if unsuccessful */
: "+m"(*l), "=&r"(tmp1), "=&r"(tmp2)
: "r"(l)
: "cc");
while (atomic_flag_test_and_set_explicit(&l->v, memory_order_acquire)) {
/* do nothing */
}
}

static inline bool sl_try_lock(struct spinlock *l)
{
return !atomic_flag_test_and_set_explicit((volatile atomic_flag *)&l->v, memory_order_acquire);
return !atomic_flag_test_and_set_explicit(&l->v, memory_order_acquire);
}

static inline void sl_unlock(struct spinlock *l)
{
/*
* Store zero to lock's value with release semantics. This triggers an
* event which wakes up other threads waiting on a lock (no SEV needed).
*/
__asm__ volatile("stlr wzr, [%1]" : "=m"(*l) : "r"(l) : "cc");
atomic_flag_clear_explicit(&l->v, memory_order_release);
}
3 changes: 2 additions & 1 deletion src/load.c
Original file line number Diff line number Diff line change
Expand Up @@ -219,7 +219,8 @@ static bool load_primary(struct mm_stage1_locked stage1_locked,
vm->vcpu_count, pa_addr(primary_begin));

vcpu_execution_locked = vcpu_lock(vm_get_vcpu(vm, 0));
vcpu_on(vcpu_execution_locked, ipa_from_pa(primary_begin), params->kernel_arg);
vcpu_on(vcpu_execution_locked, ipa_from_pa(primary_begin),
params->kernel_arg);
vcpu_unlock(&vcpu_execution_locked);
ret = true;

Expand Down
13 changes: 13 additions & 0 deletions src/vcpu.c
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,19 @@ void vcpu_unlock(struct vcpu_execution_locked *locked)
locked->vcpu = NULL;
}

/**
* Tries to lock the given vCPU, but doesn't block if it is locked.
**/
bool vcpu_try_lock(struct vcpu *vcpu, struct vcpu_execution_locked *locked)
{
if (!sl_try_lock(&vcpu->execution_lock)) {
return false;
}

locked->vcpu = vcpu;
return true;
}

void vcpu_init(struct vcpu *vcpu, struct vm *vm)
{
memset_s(vcpu, sizeof(*vcpu), 0, sizeof(*vcpu));
Expand Down

0 comments on commit 1203c38

Please sign in to comment.