mirror of
https://github.com/switchbrew/libnx.git
synced 2025-06-21 12:32:40 +02:00
Initial time support, timezones are not handled yet.
This commit is contained in:
parent
6ee4d389a5
commit
14263ddd78
@ -35,6 +35,7 @@ extern "C" {
|
|||||||
#include "switch/services/audout.h"
|
#include "switch/services/audout.h"
|
||||||
#include "switch/services/bsd.h"
|
#include "switch/services/bsd.h"
|
||||||
#include "switch/services/fatal.h"
|
#include "switch/services/fatal.h"
|
||||||
|
#include "switch/services/time.h"
|
||||||
#include "switch/services/usb.h"
|
#include "switch/services/usb.h"
|
||||||
#include "switch/services/hid.h"
|
#include "switch/services/hid.h"
|
||||||
#include "switch/services/irs.h"
|
#include "switch/services/irs.h"
|
||||||
|
@ -64,6 +64,7 @@ enum {
|
|||||||
LibnxError_JitUnavailable,
|
LibnxError_JitUnavailable,
|
||||||
LibnxError_WeirdKernel,
|
LibnxError_WeirdKernel,
|
||||||
LibnxError_IncompatSysVer,
|
LibnxError_IncompatSysVer,
|
||||||
|
LibnxError_InitFail_Time,
|
||||||
};
|
};
|
||||||
|
|
||||||
/// libnx nvidia error codes
|
/// libnx nvidia error codes
|
||||||
|
24
nx/include/switch/services/time.h
Normal file
24
nx/include/switch/services/time.h
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
/**
|
||||||
|
* @file time.h
|
||||||
|
* @brief Time services IPC wrapper.
|
||||||
|
* @author yellows8
|
||||||
|
* @copyright libnx Authors
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "../types.h"
|
||||||
|
#include "../services/sm.h"
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
TimeType_UserSystemClock,
|
||||||
|
TimeType_NetworkSystemClock,
|
||||||
|
TimeType_LocalSystemClock,
|
||||||
|
TimeType_Default = TimeType_NetworkSystemClock,
|
||||||
|
} TimeType;
|
||||||
|
|
||||||
|
Result timeInitialize(void);
|
||||||
|
void timeExit(void);
|
||||||
|
|
||||||
|
Service* timeGetSessionService(void);
|
||||||
|
|
||||||
|
Result timeGetCurrentTime(TimeType type, u64 *timestamp);
|
@ -4,6 +4,7 @@
|
|||||||
#include "services/fatal.h"
|
#include "services/fatal.h"
|
||||||
#include "services/fs.h"
|
#include "services/fs.h"
|
||||||
#include "services/hid.h"
|
#include "services/hid.h"
|
||||||
|
#include "services/time.h"
|
||||||
#include "services/applet.h"
|
#include "services/applet.h"
|
||||||
#include "runtime/devices/fs_dev.h"
|
#include "runtime/devices/fs_dev.h"
|
||||||
|
|
||||||
@ -101,6 +102,10 @@ void __attribute__((weak)) __appInit(void)
|
|||||||
fatalSimple(MAKERESULT(Module_Libnx, LibnxError_InitFail_HID));
|
fatalSimple(MAKERESULT(Module_Libnx, LibnxError_InitFail_HID));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
rc = timeInitialize();
|
||||||
|
if (R_FAILED(rc))
|
||||||
|
fatalSimple(MAKERESULT(Module_Libnx, LibnxError_InitFail_Time));
|
||||||
|
|
||||||
rc = fsInitialize();
|
rc = fsInitialize();
|
||||||
if (R_FAILED(rc))
|
if (R_FAILED(rc))
|
||||||
fatalSimple(MAKERESULT(Module_Libnx, LibnxError_InitFail_FS));
|
fatalSimple(MAKERESULT(Module_Libnx, LibnxError_InitFail_FS));
|
||||||
@ -113,6 +118,7 @@ void __attribute__((weak)) __appExit(void)
|
|||||||
// Cleanup default services.
|
// Cleanup default services.
|
||||||
fsdevExit();
|
fsdevExit();
|
||||||
fsExit();
|
fsExit();
|
||||||
|
timeExit();
|
||||||
hidExit();
|
hidExit();
|
||||||
appletExit();
|
appletExit();
|
||||||
smExit();
|
smExit();
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
#include <sys/iosupport.h>
|
#include <sys/iosupport.h>
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
#include <sys/lock.h>
|
#include <sys/lock.h>
|
||||||
@ -8,6 +9,7 @@
|
|||||||
#include "runtime/env.h"
|
#include "runtime/env.h"
|
||||||
#include "kernel/mutex.h"
|
#include "kernel/mutex.h"
|
||||||
#include "services/fatal.h"
|
#include "services/fatal.h"
|
||||||
|
#include "services/time.h"
|
||||||
#include "result.h"
|
#include "result.h"
|
||||||
|
|
||||||
void __attribute__((weak)) NORETURN __libnx_exit(int rc);
|
void __attribute__((weak)) NORETURN __libnx_exit(int rc);
|
||||||
@ -16,6 +18,9 @@ extern const u8 __tdata_lma[];
|
|||||||
extern const u8 __tdata_lma_end[];
|
extern const u8 __tdata_lma_end[];
|
||||||
extern u8 __tls_start[];
|
extern u8 __tls_start[];
|
||||||
|
|
||||||
|
/// TimeType passed to timeGetCurrentTime() by __libnx_gtod().
|
||||||
|
__attribute__((weak)) TimeType __nx_time_type = TimeType_Default;
|
||||||
|
|
||||||
static struct _reent* __libnx_get_reent(void) {
|
static struct _reent* __libnx_get_reent(void) {
|
||||||
ThreadVars* tv = getThreadVars();
|
ThreadVars* tv = getThreadVars();
|
||||||
if (tv->magic != THREADVARS_MAGIC)
|
if (tv->magic != THREADVARS_MAGIC)
|
||||||
@ -23,9 +28,33 @@ static struct _reent* __libnx_get_reent(void) {
|
|||||||
return tv->reent;
|
return tv->reent;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int __libnx_gtod(struct _reent *ptr, struct timeval *tp, struct timezone *tz) {
|
||||||
|
if (tp != NULL) {
|
||||||
|
u64 now=0;
|
||||||
|
Result rc=0;
|
||||||
|
|
||||||
|
rc = timeGetCurrentTime(__nx_time_type, &now);
|
||||||
|
if (R_FAILED(rc)) {
|
||||||
|
ptr->_errno = EINVAL;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
tp->tv_sec = now;
|
||||||
|
tp->tv_usec = 0;//timeGetCurrentTime() only returns seconds.
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tz != NULL) {//TODO: This needs handled properly, timeGetCurrentTime() returns UTC time.
|
||||||
|
tz->tz_minuteswest = 0;
|
||||||
|
tz->tz_dsttime = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
void newlibSetup(void) {
|
void newlibSetup(void) {
|
||||||
// Register newlib syscalls
|
// Register newlib syscalls
|
||||||
__syscalls.exit = __libnx_exit;
|
__syscalls.exit = __libnx_exit;
|
||||||
|
__syscalls.gettod_r = __libnx_gtod;
|
||||||
__syscalls.getreent = __libnx_get_reent;
|
__syscalls.getreent = __libnx_get_reent;
|
||||||
|
|
||||||
// Register locking syscalls
|
// Register locking syscalls
|
||||||
|
143
nx/source/services/time.c
Normal file
143
nx/source/services/time.c
Normal file
@ -0,0 +1,143 @@
|
|||||||
|
#include <string.h>
|
||||||
|
#include "types.h"
|
||||||
|
#include "result.h"
|
||||||
|
#include "ipc.h"
|
||||||
|
#include "services/time.h"
|
||||||
|
#include "services/sm.h"
|
||||||
|
|
||||||
|
static Service g_timeSrv;
|
||||||
|
static Service g_timeUserSystemClock;
|
||||||
|
static Service g_timeNetworkSystemClock;
|
||||||
|
static Service g_timeTimeZoneService;
|
||||||
|
static Service g_timeLocalSystemClock;
|
||||||
|
|
||||||
|
static Result _timeGetSession(Service* srv_out, u64 cmd_id);
|
||||||
|
|
||||||
|
Result timeInitialize(void)
|
||||||
|
{
|
||||||
|
if (serviceIsActive(&g_timeSrv))
|
||||||
|
return MAKERESULT(Module_Libnx, LibnxError_AlreadyInitialized);
|
||||||
|
|
||||||
|
Result rc;
|
||||||
|
|
||||||
|
rc = smGetService(&g_timeSrv, "time:u");
|
||||||
|
if (R_FAILED(rc))
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
rc = _timeGetSession(&g_timeUserSystemClock, 0);
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(rc))
|
||||||
|
rc = _timeGetSession(&g_timeNetworkSystemClock, 1);
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(rc))
|
||||||
|
rc = _timeGetSession(&g_timeTimeZoneService, 3);
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(rc))
|
||||||
|
rc = _timeGetSession(&g_timeLocalSystemClock, 4);
|
||||||
|
|
||||||
|
if (R_FAILED(rc))
|
||||||
|
timeExit();
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
void timeExit(void)
|
||||||
|
{
|
||||||
|
if (!serviceIsActive(&g_timeSrv))
|
||||||
|
return;
|
||||||
|
|
||||||
|
serviceClose(&g_timeLocalSystemClock);
|
||||||
|
serviceClose(&g_timeTimeZoneService);
|
||||||
|
serviceClose(&g_timeNetworkSystemClock);
|
||||||
|
serviceClose(&g_timeUserSystemClock);
|
||||||
|
serviceClose(&g_timeSrv);
|
||||||
|
}
|
||||||
|
|
||||||
|
Service* timeGetSessionService(void) {
|
||||||
|
return &g_timeSrv;
|
||||||
|
}
|
||||||
|
|
||||||
|
static Result _timeGetSession(Service* srv_out, u64 cmd_id) {
|
||||||
|
IpcCommand c;
|
||||||
|
ipcInitialize(&c);
|
||||||
|
|
||||||
|
struct {
|
||||||
|
u64 magic;
|
||||||
|
u64 cmd_id;
|
||||||
|
} *raw;
|
||||||
|
|
||||||
|
raw = ipcPrepareHeader(&c, sizeof(*raw));
|
||||||
|
|
||||||
|
raw->magic = SFCI_MAGIC;
|
||||||
|
raw->cmd_id = cmd_id;
|
||||||
|
|
||||||
|
Result rc = serviceIpcDispatch(&g_timeSrv);
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(rc)) {
|
||||||
|
IpcParsedCommand r;
|
||||||
|
ipcParse(&r);
|
||||||
|
|
||||||
|
struct {
|
||||||
|
u64 magic;
|
||||||
|
u64 result;
|
||||||
|
} *resp = r.Raw;
|
||||||
|
|
||||||
|
rc = resp->result;
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(rc)) {
|
||||||
|
serviceCreate(srv_out, r.Handles[0]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
Result timeGetCurrentTime(TimeType type, u64 *timestamp) {
|
||||||
|
Service *srv = NULL;
|
||||||
|
|
||||||
|
if (type==TimeType_UserSystemClock) {
|
||||||
|
srv = &g_timeUserSystemClock;
|
||||||
|
}
|
||||||
|
else if (type==TimeType_NetworkSystemClock) {
|
||||||
|
srv = &g_timeNetworkSystemClock;
|
||||||
|
}
|
||||||
|
else if (type==TimeType_LocalSystemClock) {
|
||||||
|
srv = &g_timeLocalSystemClock;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return MAKERESULT(Module_Libnx, LibnxError_BadInput);
|
||||||
|
}
|
||||||
|
|
||||||
|
IpcCommand c;
|
||||||
|
ipcInitialize(&c);
|
||||||
|
|
||||||
|
struct {
|
||||||
|
u64 magic;
|
||||||
|
u64 cmd_id;
|
||||||
|
} *raw;
|
||||||
|
|
||||||
|
raw = ipcPrepareHeader(&c, sizeof(*raw));
|
||||||
|
|
||||||
|
raw->magic = SFCI_MAGIC;
|
||||||
|
raw->cmd_id = 0;
|
||||||
|
|
||||||
|
Result rc = serviceIpcDispatch(srv);
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(rc)) {
|
||||||
|
IpcParsedCommand r;
|
||||||
|
ipcParse(&r);
|
||||||
|
|
||||||
|
struct {
|
||||||
|
u64 magic;
|
||||||
|
u64 result;
|
||||||
|
u64 timestamp;
|
||||||
|
} *resp = r.Raw;
|
||||||
|
|
||||||
|
rc = resp->result;
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(rc) && timestamp) *timestamp = resp->timestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user