libnx/nx/source/services/pl.c

167 lines
3.8 KiB
C

#include <string.h>
#include "types.h"
#include "result.h"
#include "arm/atomics.h"
#include "kernel/ipc.h"
#include "kernel/shmem.h"
#include "services/sm.h"
#include "services/pl.h"
#define SHAREDMEMFONT_SIZE 0x1100000
static Service g_plSrv;
static u64 g_plRefCnt;
static SharedMemory g_plSharedmem;
static Result _plGetSharedMemoryNativeHandle(Handle* handle_out);
Result plInitialize(void)
{
Result rc=0;
Handle sharedmem_handle=0;
atomicIncrement64(&g_plRefCnt);
if (serviceIsActive(&g_plSrv))
return 0;
rc = smGetService(&g_plSrv, "pl:u");
if (R_SUCCEEDED(rc))
{
rc = _plGetSharedMemoryNativeHandle(&sharedmem_handle);
if (R_SUCCEEDED(rc))
{
shmemLoadRemote(&g_plSharedmem, sharedmem_handle, 0x1100000, Perm_R);
rc = shmemMap(&g_plSharedmem);
}
}
if (R_FAILED(rc)) plExit();
return rc;
}
void plExit(void)
{
if (atomicDecrement64(&g_plRefCnt) == 0) {
serviceClose(&g_plSrv);
shmemClose(&g_plSharedmem);
}
}
void* plGetSharedmemAddr(void) {
return shmemGetAddr(&g_plSharedmem);
}
static Result _plGetSharedMemoryNativeHandle(Handle* handle_out) {
IpcCommand c;
ipcInitialize(&c);
struct {
u64 magic;
u64 cmd_id;
} *raw;
raw = ipcPrepareHeader(&c, sizeof(*raw));
raw->magic = SFCI_MAGIC;
raw->cmd_id = 4;
Result rc = serviceIpcDispatch(&g_plSrv);
if (R_SUCCEEDED(rc)) {
IpcParsedCommand r;
ipcParse(&r);
struct {
u64 magic;
u64 result;
} *resp = r.Raw;
rc = resp->result;
if (R_SUCCEEDED(rc)) {
*handle_out = r.Handles[0];
}
}
return rc;
}
Result plGetSharedFont(u64 LanguageCode, PlFontData* fonts, size_t max_fonts, size_t* total_fonts) {
u32 types[PlSharedFontType_Total];
u32 offsets[PlSharedFontType_Total];
u32 sizes[PlSharedFontType_Total];
size_t size = sizeof(u32) * PlSharedFontType_Total;
u32 font_count=0, i;
u8* sharedmem_addr = (u8*)plGetSharedmemAddr();
IpcCommand c;
ipcInitialize(&c);
ipcAddRecvBuffer(&c, types, size, 0);
ipcAddRecvBuffer(&c, offsets, size, 0);
ipcAddRecvBuffer(&c, sizes, size, 0);
memset(types, 0, sizeof(types));
memset(offsets, 0, sizeof(offsets));
memset(sizes, 0, sizeof(sizes));
memset(fonts, 0, sizeof(PlFontData) * max_fonts);
if (total_fonts) *total_fonts = 0;
struct {
u64 magic;
u64 cmd_id;
u64 LanguageCode;
} *raw;
raw = ipcPrepareHeader(&c, sizeof(*raw));
raw->magic = SFCI_MAGIC;
raw->cmd_id = 5;
raw->LanguageCode = LanguageCode;
Result rc = serviceIpcDispatch(&g_plSrv);
if (R_SUCCEEDED(rc)) {
IpcParsedCommand r;
ipcParse(&r);
struct {
u64 magic;
u64 result;
u8 fonts_loaded;
u32 total_fonts;
} *resp = r.Raw;
rc = resp->result;
if (R_SUCCEEDED(rc)) {
if (resp->fonts_loaded==0) return rc;
font_count = resp->total_fonts;
if (font_count > PlSharedFontType_Total) font_count = PlSharedFontType_Total;
if (font_count > max_fonts) font_count = max_fonts;
if (total_fonts) *total_fonts = font_count;
if (font_count==0) return rc;
for (i=0; i<font_count; i++) {
if (offsets[i] >= SHAREDMEMFONT_SIZE || sizes[i] > SHAREDMEMFONT_SIZE || ((u64)offsets[i] + (u64)sizes[i]) > SHAREDMEMFONT_SIZE)
return MAKERESULT(Module_Libnx, LibnxError_BadInput);
fonts[i].type = types[i];
fonts[i].offset = offsets[i];
fonts[i].size = sizes[i];
fonts[i].address = &sharedmem_addr[offsets[i]];
}
}
}
return rc;
}