2020-12-05 13:08:24 +01:00
|
|
|
#pragma once
|
2014-08-08 19:55:12 +04:00
|
|
|
|
2016-04-14 01:23:53 +03:00
|
|
|
#include "sys_sync.h"
|
2015-06-19 18:49:38 +03:00
|
|
|
|
2018-09-26 01:14:10 +03:00
|
|
|
#include "Emu/Memory/vm_ptr.h"
|
|
|
|
|
|
2014-08-08 19:55:12 +04:00
|
|
|
enum
|
|
|
|
|
{
|
|
|
|
|
SYS_SYNC_WAITER_SINGLE = 0x10000,
|
|
|
|
|
SYS_SYNC_WAITER_MULTIPLE = 0x20000,
|
|
|
|
|
|
|
|
|
|
SYS_EVENT_FLAG_WAIT_AND = 0x01,
|
|
|
|
|
SYS_EVENT_FLAG_WAIT_OR = 0x02,
|
|
|
|
|
|
|
|
|
|
SYS_EVENT_FLAG_WAIT_CLEAR = 0x10,
|
|
|
|
|
SYS_EVENT_FLAG_WAIT_CLEAR_ALL = 0x20,
|
|
|
|
|
};
|
|
|
|
|
|
2015-07-09 18:30:37 +03:00
|
|
|
struct sys_event_flag_attribute_t
|
2014-08-08 19:55:12 +04:00
|
|
|
{
|
|
|
|
|
be_t<u32> protocol;
|
|
|
|
|
be_t<u32> pshared;
|
|
|
|
|
be_t<u64> ipc_key;
|
2014-12-24 19:09:32 +03:00
|
|
|
be_t<s32> flags;
|
|
|
|
|
be_t<s32> type;
|
2015-03-06 00:29:05 +03:00
|
|
|
|
2014-12-24 19:09:32 +03:00
|
|
|
union
|
|
|
|
|
{
|
2019-10-25 17:50:46 +03:00
|
|
|
nse_t<u64, 1> name_u64;
|
2019-07-08 20:00:27 +03:00
|
|
|
char name[sizeof(u64)];
|
2014-12-24 19:09:32 +03:00
|
|
|
};
|
2014-08-08 19:55:12 +04:00
|
|
|
};
|
|
|
|
|
|
2017-01-29 19:50:18 +03:00
|
|
|
struct lv2_event_flag final : lv2_obj
|
2014-08-08 19:55:12 +04:00
|
|
|
{
|
2017-01-25 20:50:30 +03:00
|
|
|
static const u32 id_base = 0x98000000;
|
|
|
|
|
|
2020-06-09 18:35:14 +03:00
|
|
|
const lv2_protocol protocol;
|
2017-02-03 02:16:09 +03:00
|
|
|
const u64 key;
|
2015-01-13 17:54:36 +03:00
|
|
|
const s32 type;
|
2014-12-24 19:09:32 +03:00
|
|
|
const u64 name;
|
2014-08-08 19:55:12 +04:00
|
|
|
|
2018-11-26 18:55:22 +03:00
|
|
|
shared_mutex mutex;
|
2016-04-14 01:23:53 +03:00
|
|
|
atomic_t<u64> pattern;
|
2022-07-28 14:10:16 +03:00
|
|
|
ppu_thread* sq{};
|
2015-03-06 00:29:05 +03:00
|
|
|
|
2021-05-08 20:08:25 +03:00
|
|
|
lv2_event_flag(u32 protocol, u64 key, s32 type, u64 name, u64 pattern) noexcept
|
|
|
|
|
: protocol{static_cast<u8>(protocol)}
|
2017-07-24 18:59:48 +03:00
|
|
|
, key(key)
|
2014-12-23 02:31:11 +03:00
|
|
|
, type(type)
|
2014-12-24 19:09:32 +03:00
|
|
|
, name(name)
|
2017-02-03 02:16:09 +03:00
|
|
|
, pattern(pattern)
|
2014-08-08 19:55:12 +04:00
|
|
|
{
|
|
|
|
|
}
|
2015-07-20 00:29:40 +03:00
|
|
|
|
2022-07-04 16:02:17 +03:00
|
|
|
lv2_event_flag(utils::serial& ar);
|
|
|
|
|
static std::shared_ptr<void> load(utils::serial& ar);
|
|
|
|
|
void save(utils::serial& ar);
|
|
|
|
|
|
2017-02-03 19:27:03 +03:00
|
|
|
// Check mode arg
|
|
|
|
|
static bool check_mode(u32 mode)
|
2015-07-20 00:29:40 +03:00
|
|
|
{
|
|
|
|
|
switch (mode & 0xf)
|
|
|
|
|
{
|
|
|
|
|
case SYS_EVENT_FLAG_WAIT_AND: break;
|
|
|
|
|
case SYS_EVENT_FLAG_WAIT_OR: break;
|
|
|
|
|
default: return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (mode & ~0xf)
|
|
|
|
|
{
|
|
|
|
|
case 0: break;
|
|
|
|
|
case SYS_EVENT_FLAG_WAIT_CLEAR: break;
|
|
|
|
|
case SYS_EVENT_FLAG_WAIT_CLEAR_ALL: break;
|
|
|
|
|
default: return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-03 19:27:03 +03:00
|
|
|
// Check and clear pattern (must be atomic op)
|
|
|
|
|
static bool check_pattern(u64& pattern, u64 bitptn, u64 mode, u64* result)
|
2015-07-20 00:29:40 +03:00
|
|
|
{
|
2017-02-03 19:27:03 +03:00
|
|
|
// Write pattern
|
|
|
|
|
if (result)
|
2015-07-20 00:29:40 +03:00
|
|
|
{
|
2017-02-03 19:27:03 +03:00
|
|
|
*result = pattern;
|
2015-07-20 00:29:40 +03:00
|
|
|
}
|
2017-02-03 19:27:03 +03:00
|
|
|
|
|
|
|
|
// Check pattern
|
2017-11-01 23:28:40 -07:00
|
|
|
if (((mode & 0xf) == SYS_EVENT_FLAG_WAIT_AND && (pattern & bitptn) != bitptn) ||
|
|
|
|
|
((mode & 0xf) == SYS_EVENT_FLAG_WAIT_OR && (pattern & bitptn) == 0))
|
2015-07-20 00:29:40 +03:00
|
|
|
{
|
2017-02-03 19:27:03 +03:00
|
|
|
return false;
|
2015-07-20 00:29:40 +03:00
|
|
|
}
|
|
|
|
|
|
2017-02-03 19:27:03 +03:00
|
|
|
// Clear pattern if necessary
|
2015-07-20 00:29:40 +03:00
|
|
|
if ((mode & ~0xf) == SYS_EVENT_FLAG_WAIT_CLEAR)
|
|
|
|
|
{
|
2017-02-03 19:27:03 +03:00
|
|
|
pattern &= ~bitptn;
|
2015-07-20 00:29:40 +03:00
|
|
|
}
|
|
|
|
|
else if ((mode & ~0xf) == SYS_EVENT_FLAG_WAIT_CLEAR_ALL)
|
|
|
|
|
{
|
2017-02-03 19:27:03 +03:00
|
|
|
pattern = 0;
|
2015-07-20 00:29:40 +03:00
|
|
|
}
|
|
|
|
|
|
2017-02-03 19:27:03 +03:00
|
|
|
return true;
|
|
|
|
|
}
|
2014-08-08 19:55:12 +04:00
|
|
|
};
|
|
|
|
|
|
2015-07-20 00:29:40 +03:00
|
|
|
// Aux
|
2016-07-28 00:43:22 +03:00
|
|
|
class ppu_thread;
|
2015-07-20 00:29:40 +03:00
|
|
|
|
2017-02-03 19:27:03 +03:00
|
|
|
// Syscalls
|
|
|
|
|
|
2019-07-14 17:37:58 +03:00
|
|
|
error_code sys_event_flag_create(ppu_thread& ppu, vm::ptr<u32> id, vm::ptr<sys_event_flag_attribute_t> attr, u64 init);
|
|
|
|
|
error_code sys_event_flag_destroy(ppu_thread& ppu, u32 id);
|
2018-02-09 17:49:37 +03:00
|
|
|
error_code sys_event_flag_wait(ppu_thread& ppu, u32 id, u64 bitptn, u32 mode, vm::ptr<u64> result, u64 timeout);
|
2019-07-14 17:37:58 +03:00
|
|
|
error_code sys_event_flag_trywait(ppu_thread& ppu, u32 id, u64 bitptn, u32 mode, vm::ptr<u64> result);
|
2020-10-30 16:32:49 +03:00
|
|
|
error_code sys_event_flag_set(cpu_thread& cpu, u32 id, u64 bitptn);
|
2019-07-14 17:37:58 +03:00
|
|
|
error_code sys_event_flag_clear(ppu_thread& ppu, u32 id, u64 bitptn);
|
2018-02-09 17:49:37 +03:00
|
|
|
error_code sys_event_flag_cancel(ppu_thread& ppu, u32 id, vm::ptr<u32> num);
|
2019-07-14 17:37:58 +03:00
|
|
|
error_code sys_event_flag_get(ppu_thread& ppu, u32 id, vm::ptr<u64> flags);
|