/* * BrainLILODrv * U-Boot loader for electric dictionary. * * Copyright (C) 2019 C. Shirasaka * based on ** ResetKitHelper ** Soft/hard reset the electronic dictionary. ** ** Copyright (C) 2012 T. Kawada * * This file is licensed in MIT license. * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation the * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or * sell copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. * */ #include #include #include #include #include #define FSNOTIFY_POWER_OFF 1 #define FSNOTIFY_POWER_ON 0 #define BRAINLILODRV_API __declspec(dllexport) #include "BrainLILODrv.h" #define FILE_DEVICE_POWER FILE_DEVICE_ACPI #define IOCTL_POWER_CAPABILITIES CTL_CODE(FILE_DEVICE_POWER, 0x400, METHOD_BUFFERED, FILE_ANY_ACCESS) #define IOCTL_POWER_GET CTL_CODE(FILE_DEVICE_POWER, 0x401, METHOD_BUFFERED, FILE_ANY_ACCESS) #define IOCTL_POWER_SET CTL_CODE(FILE_DEVICE_POWER, 0x402, METHOD_BUFFERED, FILE_ANY_ACCESS) #define IOCTL_POWER_QUERY CTL_CODE(FILE_DEVICE_POWER, 0x403, METHOD_BUFFERED, FILE_ANY_ACCESS) typedef BOOL (*KernelIoControlProc)(DWORD dwIoControlCode, LPVOID lpInBuf, DWORD nInBufSize, LPVOID lpOutBuf, DWORD nOutBufSize, LPDWORD lpBytesReturned); static KernelIoControlProc KernelIoControl; typedef LARGE_INTEGER PHYSICAL_ADDRESS, *PPHYSICAL_ADDRESS; typedef PVOID (*MmMapIoSpaceProc)(PHYSICAL_ADDRESS, ULONG, BOOL); static MmMapIoSpaceProc MmMapIoSpace; typedef void (*FileSystemPowerFunctionProc)(DWORD); static FileSystemPowerFunctionProc FileSystemPowerFunction; typedef LPVOID (*AllocPhysMemProc)(DWORD, DWORD, DWORD, DWORD, PULONG); DWORD FileSize; static void disableInterrupts() { asm volatile("mrs r0, cpsr\n" "orr r0,r0,#0x80\n" "msr cpsr_c,r0\n" "mov r0,#1" :: : "r0"); } static void EDNA2_physicalInvoker() { // r0-r7=params // r8=proc address asm volatile("nop\n" // who cares interrupt vectors? "nop\n" "nop\n" "nop\n" "nop\n" "nop\n" "nop\n" "nop\n" "nop\n" "nop\n" "msr cpsr_c, #211\n" // to supervisor mode "mov r9, #0\n" "mcr p15,0,r9,c13,c0,0\n" // clear fcse PID "mrc p15,0,r9,c1,c0,0\n" // read ctrl regs "bic r9, r9, #5\n" // disable MMU/DCache "bic r9, r9, #4096\n" // disable ICache "orr r9, r9, #8192\n" // and reset vectors to upper "mcr p15,0,r9,c1,c0,0\n" // write ctrl regs "mov r9, #0\n" "mcr p15,0,r9,c7,c7,0\n" // invalidate cache "mcr p15,0,r9,c8,c7,0\n" // invalidate tlb "mov pc, r8\n" "nop\n" "nop\n"); } static void EDNA2_installPhysicalInvoker() { void *ptr = (void *)0xa8000000; wchar_t buf[256]; swprintf(buf, L"BrainLILO: copying PhysicalInvoker to 0x%08x from 0x%08x\n", (int)(ptr), (int)(&EDNA2_physicalInvoker)); OutputDebugString(buf); memcpy(ptr, (const void *)&EDNA2_physicalInvoker, 64 * 4); // clearCache(); } __attribute__((noreturn)) static void EDNA2_runPhysicalInvoker() { // r0=info asm volatile("msr cpsr_c, #211\n" // to supervisor mode "mrc p15,0,r0,c1,c0,0\n" // read ctrl regs "bic r0, r0, #8192\n" // reset vector to lower "mcr p15,0,r0,c1,c0,0\n" // write ctrl regs ); for (DWORD i = 0; i < FileSize; i++) *((char *)(0xa0200000 + i)) = *((char *)(0xa0000000 + i)); asm volatile("ldr r0, =0x0000\n" "ldr r1, =0x0000\n" "ldr r2, =0x0000\n" "ldr r3, =0x0000\n" "ldr r4, =0x0000\n" "ldr r5, =0x0000\n" "ldr r6, =0x0000\n" "ldr r7, =0x0000\n" "ldr r8, =0x40200000\n" "ldr r9, =0x0000\n" "mrc p15,0,r10,c1,c0,0\n" // read ctrl regs "bic r10, r10, #5\n" // disable MMU/DCache "mcr p15,0,r10,c1,c0,0\n" // write ctrl regs "swi #0\n" // jump! ); // never reach here while (true) ; } __attribute__((noreturn)) static DWORD EDNA2_callKernelEntryPoint() { OutputDebugString(L"BrainLILO: disabling interrupts"); disableInterrupts(); OutputDebugString(L"BrainLILO: injecting code to internal ram"); EDNA2_installPhysicalInvoker(); OutputDebugString(L"BrainLILO: invoking"); EDNA2_runPhysicalInvoker(); } static void ShowMessage(std::string msg, std::string title, UINT typ) { void *bufMsg; void *bufTitle; bufMsg = LocalAlloc(LPTR, msg.length() * sizeof(wchar_t)); bufTitle = LocalAlloc(LPTR, title.length() * sizeof(wchar_t)); mbstowcs((wchar_t *)bufMsg, msg.c_str(), msg.length()); mbstowcs((wchar_t *)bufTitle, title.c_str(), title.length()); MessageBox(NULL, (wchar_t *)bufMsg, (wchar_t *)bufTitle, typ); LocalFree(bufMsg); LocalFree(bufTitle); } static bool doLinux() { wchar_t wcBuf[256] = {}; std::ifstream iVersion; std::string line, model; std::regex modelRe("[A-Z]{2}-[A-Z0-9]+"); std::smatch match; std::string fn("\\Storage Card\\loader\\"); HANDLE hUBoot; DWORD wReadSize; iVersion.open("\\NAND\\version.txt"); while (getline(iVersion, line)) { if (regex_search(line, match, modelRe)) { model = match[0].str(); break; } } if (model.length() == 0) { ShowMessage("Failed to match the model name", "BrainLILO", MB_ICONWARNING); return false; } OutputDebugString(L"BrainLILO: Opening Bootloader file..."); fn += model + ".bin"; mbstowcs(wcBuf, fn.c_str(), fn.length()); hUBoot = CreateFile(wcBuf, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (hUBoot == INVALID_HANDLE_VALUE) { OutputDebugString(L"Could not open the bootloader"); ShowMessage("Could not open the bootloader: " + fn, "BrainLILO", MB_ICONWARNING); return false; } swprintf(wcBuf, L"BrainLILO: Bootloader file handle 0x%08x\n", (int)(hUBoot)); OutputDebugString(wcBuf); FileSize = GetFileSize(hUBoot, NULL); swprintf(wcBuf, L"BrainLILO: Bootloader file size %d Byte\n", FileSize); OutputDebugString(wcBuf); OutputDebugString(L"BrainLILO: Preloading bootloader to 0xa0000000..."); if (!ReadFile(hUBoot, (void *)0xa0000000, FileSize, &wReadSize, NULL)) { OutputDebugString(L"Could not read the bootloader"); ShowMessage("Could not read the bootloader", "BrainLILO", MB_ICONWARNING); return false; } OutputDebugString(L"BrainLILO: Bootloader copied! Closing file handle..."); CloseHandle(hUBoot); OutputDebugString(L"BrainLILO: Notifying power off to filesystems..."); if (FileSystemPowerFunction) FileSystemPowerFunction(FSNOTIFY_POWER_OFF); OutputDebugString(L"BrainLILO: Starting bootloader call sequence..."); EDNA2_callKernelEntryPoint(); return true; } extern "C" BRAINLILODRV_API BOOL LIN_IOControl(DWORD handle, DWORD dwIoControlCode, DWORD *pInBuf, DWORD nInBufSize, DWORD *pOutBuf, DWORD nOutBufSize, PDWORD pBytesReturned) { SetLastError(0); switch (dwIoControlCode) { case IOCTL_LIN_DO_LINUX: if (!doLinux()) { if (FileSystemPowerFunction) FileSystemPowerFunction(FSNOTIFY_POWER_ON); return FALSE; } return TRUE; } return FALSE; } extern "C" BRAINLILODRV_API BOOL LIN_Read(DWORD handle, LPVOID pBuffer, DWORD dwNumBytes) { SetLastError(ERROR_INVALID_FUNCTION); return FALSE; } extern "C" BRAINLILODRV_API BOOL LIN_Write(DWORD handle, LPVOID pBuffer, DWORD dwNumBytes) { SetLastError(ERROR_INVALID_FUNCTION); return FALSE; } extern "C" BRAINLILODRV_API DWORD LIN_Seek(DWORD handle, long lDistance, DWORD dwMoveMethod) { SetLastError(ERROR_INVALID_FUNCTION); return FALSE; } extern "C" BRAINLILODRV_API void LIN_PowerUp(void) { OutputDebugString(L"BrainLILO: resuming."); } extern "C" BRAINLILODRV_API void LIN_PowerDown(void) { } extern "C" BRAINLILODRV_API DWORD LIN_Init(LPCTSTR pContext, DWORD dwBusContext) { void *ctx; ctx = (void *)LocalAlloc(LPTR, sizeof(4)); return (DWORD)ctx; } extern "C" BRAINLILODRV_API DWORD LIN_Open(DWORD dwData, DWORD dwAccess, DWORD dwShareMode) { void *hnd = (void *)LocalAlloc(LPTR, 4); return (DWORD)hnd; } extern "C" BRAINLILODRV_API BOOL LIN_Close(DWORD handle) { LocalFree((void *)handle); return TRUE; } extern "C" BRAINLILODRV_API BOOL LIN_Deinit(DWORD dwContext) { LocalFree((void *)dwContext); return TRUE; } extern "C" BOOL APIENTRY DllMainCRTStartup(HANDLE hModule, DWORD ul_reason_for_call, LPVOID lpReserved) { switch (ul_reason_for_call) { case DLL_PROCESS_ATTACH: case DLL_THREAD_ATTACH: KernelIoControl = (KernelIoControlProc)GetProcAddress(LoadLibrary(L"COREDLL"), L"KernelIoControl"); MmMapIoSpace = (MmMapIoSpaceProc)GetProcAddress(LoadLibrary(L"CEDDK"), L"MmMapIoSpace"); FileSystemPowerFunction = (FileSystemPowerFunctionProc)GetProcAddress(LoadLibrary(L"COREDLL"), L"FileSystemPowerFunction"); case DLL_THREAD_DETACH: case DLL_PROCESS_DETACH: break; } return TRUE; }