2014-08-08 17:55:12 +02:00
|
|
|
#pragma once
|
|
|
|
|
|
2016-04-14 00:23:53 +02:00
|
|
|
#include "sys_sync.h"
|
2015-06-19 17:49:38 +02:00
|
|
|
|
2014-08-08 17:55:12 +02: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 17:30:37 +02:00
|
|
|
struct sys_event_flag_attribute_t
|
2014-08-08 17:55:12 +02:00
|
|
|
{
|
|
|
|
|
be_t<u32> protocol;
|
|
|
|
|
be_t<u32> pshared;
|
|
|
|
|
be_t<u64> ipc_key;
|
2014-12-24 17:09:32 +01:00
|
|
|
be_t<s32> flags;
|
|
|
|
|
be_t<s32> type;
|
2015-03-05 22:29:05 +01:00
|
|
|
|
2014-12-24 17:09:32 +01:00
|
|
|
union
|
|
|
|
|
{
|
|
|
|
|
char name[8];
|
|
|
|
|
u64 name_u64;
|
|
|
|
|
};
|
2014-08-08 17:55:12 +02:00
|
|
|
};
|
|
|
|
|
|
2017-01-29 17:50:18 +01:00
|
|
|
struct lv2_event_flag final : lv2_obj
|
2014-08-08 17:55:12 +02:00
|
|
|
{
|
2017-01-25 18:50:30 +01:00
|
|
|
static const u32 id_base = 0x98000000;
|
|
|
|
|
|
2014-12-23 00:31:11 +01:00
|
|
|
const u32 protocol;
|
2017-02-03 00:16:09 +01:00
|
|
|
const u32 shared;
|
|
|
|
|
const u64 key;
|
|
|
|
|
const s32 flags;
|
2015-01-13 15:54:36 +01:00
|
|
|
const s32 type;
|
2014-12-24 17:09:32 +01:00
|
|
|
const u64 name;
|
2014-08-08 17:55:12 +02:00
|
|
|
|
2016-04-14 00:23:53 +02:00
|
|
|
atomic_t<u64> pattern;
|
2015-03-11 11:39:54 +01:00
|
|
|
|
2016-04-14 00:23:53 +02:00
|
|
|
sleep_queue<cpu_thread> sq;
|
2015-03-05 22:29:05 +01:00
|
|
|
|
2017-02-03 00:16:09 +01:00
|
|
|
lv2_event_flag(u32 protocol, s32 type, u64 name, u64 pattern)
|
|
|
|
|
: protocol(protocol)
|
|
|
|
|
, shared(0)
|
|
|
|
|
, key(0)
|
|
|
|
|
, flags(0)
|
2014-12-23 00:31:11 +01:00
|
|
|
, type(type)
|
2014-12-24 17:09:32 +01:00
|
|
|
, name(name)
|
2017-02-03 00:16:09 +01:00
|
|
|
, pattern(pattern)
|
2014-08-08 17:55:12 +02:00
|
|
|
{
|
|
|
|
|
}
|
2015-07-19 23:29:40 +02:00
|
|
|
|
|
|
|
|
static inline bool check_mode(u32 mode)
|
|
|
|
|
{
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-26 22:46:04 +02:00
|
|
|
bool check_pattern(u64 bitptn, u32 mode)
|
2015-07-19 23:29:40 +02:00
|
|
|
{
|
|
|
|
|
if ((mode & 0xf) == SYS_EVENT_FLAG_WAIT_AND)
|
|
|
|
|
{
|
|
|
|
|
return (pattern & bitptn) == bitptn;
|
|
|
|
|
}
|
|
|
|
|
else if ((mode & 0xf) == SYS_EVENT_FLAG_WAIT_OR)
|
|
|
|
|
{
|
|
|
|
|
return (pattern & bitptn) != 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2016-08-08 18:01:06 +02:00
|
|
|
fmt::throw_exception("Unknown mode (0x%x)" HERE, mode);
|
2015-07-19 23:29:40 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-26 22:46:04 +02:00
|
|
|
u64 clear_pattern(u64 bitptn, u32 mode)
|
2015-07-19 23:29:40 +02:00
|
|
|
{
|
|
|
|
|
if ((mode & ~0xf) == SYS_EVENT_FLAG_WAIT_CLEAR)
|
|
|
|
|
{
|
|
|
|
|
return pattern.fetch_and(~bitptn);
|
|
|
|
|
}
|
|
|
|
|
else if ((mode & ~0xf) == SYS_EVENT_FLAG_WAIT_CLEAR_ALL)
|
|
|
|
|
{
|
|
|
|
|
return pattern.exchange(0);
|
|
|
|
|
}
|
|
|
|
|
else if ((mode & ~0xf) == 0)
|
|
|
|
|
{
|
|
|
|
|
return pattern;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2016-08-08 18:01:06 +02:00
|
|
|
fmt::throw_exception("Unknown mode (0x%x)" HERE, mode);
|
2015-07-19 23:29:40 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-14 00:23:53 +02:00
|
|
|
void notify_all(lv2_lock_t);
|
2014-08-08 17:55:12 +02:00
|
|
|
};
|
|
|
|
|
|
2015-07-19 23:29:40 +02:00
|
|
|
// Aux
|
2016-07-27 23:43:22 +02:00
|
|
|
class ppu_thread;
|
2015-07-19 23:29:40 +02:00
|
|
|
|
|
|
|
|
// SysCalls
|
2016-08-19 23:14:10 +02:00
|
|
|
s32 sys_event_flag_create(vm::ps3::ptr<u32> id, vm::ps3::ptr<sys_event_flag_attribute_t> attr, u64 init);
|
2015-03-05 22:29:05 +01:00
|
|
|
s32 sys_event_flag_destroy(u32 id);
|
2016-08-19 23:14:10 +02:00
|
|
|
s32 sys_event_flag_wait(ppu_thread& ppu, u32 id, u64 bitptn, u32 mode, vm::ps3::ptr<u64> result, u64 timeout);
|
|
|
|
|
s32 sys_event_flag_trywait(u32 id, u64 bitptn, u32 mode, vm::ps3::ptr<u64> result);
|
2015-03-05 22:29:05 +01:00
|
|
|
s32 sys_event_flag_set(u32 id, u64 bitptn);
|
|
|
|
|
s32 sys_event_flag_clear(u32 id, u64 bitptn);
|
2016-08-19 23:14:10 +02:00
|
|
|
s32 sys_event_flag_cancel(u32 id, vm::ps3::ptr<u32> num);
|
|
|
|
|
s32 sys_event_flag_get(u32 id, vm::ps3::ptr<u64> flags);
|