mirror of
https://github.com/LearningOS/rust-based-os-comp2022.git
synced 2026-05-05 10:04:21 +08:00
77 lines
1.7 KiB
Rust
77 lines
1.7 KiB
Rust
#![no_std]
|
|
#![no_main]
|
|
#![allow(clippy::println_empty_string)]
|
|
|
|
#[macro_use]
|
|
extern crate user_lib;
|
|
extern crate alloc;
|
|
|
|
use user_lib::{
|
|
enable_deadlock_detect, exit, semaphore_create, semaphore_down, semaphore_up, sleep,
|
|
};
|
|
use user_lib::{gettid, thread_create, waittid};
|
|
|
|
// sem 0: used to sync child thread with main
|
|
// sem 1-3: representing some kind of resource
|
|
|
|
// 理想结果:未检测到死锁,子线程返回值均为 0
|
|
|
|
const THREAD_N: usize = 4;
|
|
const RES_TYPE: usize = 2;
|
|
const RES_NUM: [usize; RES_TYPE] = [2, 2];
|
|
const ALLOC: [usize; THREAD_N] = [2, 1, 1, 2];
|
|
const REQUEST: [Option<usize>; THREAD_N] = [Some(1), None, Some(2), None];
|
|
|
|
fn try_sem_down(sem_id: usize) {
|
|
if semaphore_down(sem_id) == -0xdead {
|
|
semaphore_up(ALLOC[(gettid() - 1) as usize]);
|
|
exit(-1);
|
|
}
|
|
}
|
|
|
|
fn deadlock_test() {
|
|
let id = (gettid() - 1) as usize;
|
|
assert_eq!(semaphore_down(ALLOC[id]), 0);
|
|
semaphore_down(0);
|
|
if let Some(sem_id) = REQUEST[id] {
|
|
try_sem_down(sem_id);
|
|
semaphore_up(sem_id);
|
|
}
|
|
semaphore_up(ALLOC[id]);
|
|
exit(0);
|
|
}
|
|
|
|
#[no_mangle]
|
|
pub fn main() -> i32 {
|
|
enable_deadlock_detect(true);
|
|
semaphore_create(THREAD_N);
|
|
for _ in 0..THREAD_N {
|
|
semaphore_down(0);
|
|
}
|
|
|
|
for n in RES_NUM {
|
|
semaphore_create(n);
|
|
}
|
|
let mut tids = [0; THREAD_N];
|
|
|
|
for i in 0..THREAD_N {
|
|
tids[i] = thread_create(deadlock_test as usize, 0) as usize;
|
|
}
|
|
|
|
sleep(1000);
|
|
for _ in 0..THREAD_N {
|
|
semaphore_up(0);
|
|
}
|
|
|
|
let mut failed = 0;
|
|
for tid in tids {
|
|
if waittid(tid) != 0 {
|
|
failed += 1;
|
|
}
|
|
}
|
|
|
|
assert_eq!(failed, 0);
|
|
println!("deadlock test semaphore 2 OK!");
|
|
0
|
|
}
|