mirror of
				https://github.com/Atmosphere-NX/Atmosphere.git
				synced 2025-10-31 11:15:51 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			158 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			158 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) Atmosphère-NX
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms and conditions of the GNU General Public License,
 | |
|  * version 2, as published by the Free Software Foundation.
 | |
|  *
 | |
|  * This program is distributed in the hope it will be useful, but WITHOUT
 | |
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 | |
|  * more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| #include <switch.h>
 | |
| #include <string.h>
 | |
| #include "ams_su.h"
 | |
| #include "service_guard.h"
 | |
| 
 | |
| static Service g_amssuSrv;
 | |
| static TransferMemory g_tmem;
 | |
| 
 | |
| NX_GENERATE_SERVICE_GUARD(amssu);
 | |
| 
 | |
| Result _amssuInitialize(void) {
 | |
|     return smGetService(&g_amssuSrv, "ams:su");
 | |
| }
 | |
| 
 | |
| void _amssuCleanup(void) {
 | |
|     serviceClose(&g_amssuSrv);
 | |
|     tmemClose(&g_tmem);
 | |
| }
 | |
| 
 | |
| Service *amssuGetServiceSession(void) {
 | |
|     return &g_amssuSrv;
 | |
| }
 | |
| 
 | |
| Result amssuGetUpdateInformation(AmsSuUpdateInformation *out, const char *path) {
 | |
|     char send_path[FS_MAX_PATH] = {0};
 | |
|     strncpy(send_path, path, FS_MAX_PATH-1);
 | |
|     send_path[FS_MAX_PATH-1] = 0;
 | |
| 
 | |
|     return serviceDispatchOut(&g_amssuSrv, 0, *out,
 | |
|         .buffer_attrs = { SfBufferAttr_In | SfBufferAttr_HipcPointer | SfBufferAttr_FixedSize },
 | |
|         .buffers      = { { send_path, FS_MAX_PATH } },
 | |
|     );
 | |
| }
 | |
| 
 | |
| Result amssuValidateUpdate(AmsSuUpdateValidationInfo *out, const char *path) {
 | |
|     char send_path[FS_MAX_PATH] = {0};
 | |
|     strncpy(send_path, path, FS_MAX_PATH-1);
 | |
|     send_path[FS_MAX_PATH-1] = 0;
 | |
| 
 | |
|     return serviceDispatchOut(&g_amssuSrv, 1, *out,
 | |
|         .buffer_attrs = { SfBufferAttr_In | SfBufferAttr_HipcPointer | SfBufferAttr_FixedSize },
 | |
|         .buffers      = { { send_path, FS_MAX_PATH } },
 | |
|     );
 | |
| }
 | |
| 
 | |
| Result amssuSetupUpdate(void *buffer, size_t size, const char *path, bool exfat) {
 | |
|     Result rc = 0;
 | |
| 
 | |
|     if (buffer == NULL) {
 | |
|         rc = tmemCreate(&g_tmem, size, Perm_None);
 | |
|     } else {
 | |
|         rc = tmemCreateFromMemory(&g_tmem, buffer, size, Perm_None);
 | |
|     }
 | |
|     if (R_FAILED(rc)) return rc;
 | |
| 
 | |
|     char send_path[FS_MAX_PATH] = {0};
 | |
|     strncpy(send_path, path, FS_MAX_PATH-1);
 | |
|     send_path[FS_MAX_PATH-1] = 0;
 | |
| 
 | |
|     const struct {
 | |
|         u8 exfat;
 | |
|         u64 size;
 | |
|     } in = { exfat, g_tmem.size };
 | |
| 
 | |
|     rc = serviceDispatchIn(&g_amssuSrv, 2, in,
 | |
|         .in_num_handles = 1,
 | |
|         .in_handles = { g_tmem.handle },
 | |
|         .buffer_attrs = { SfBufferAttr_In | SfBufferAttr_HipcPointer | SfBufferAttr_FixedSize },
 | |
|         .buffers      = { { send_path, FS_MAX_PATH } },
 | |
|     );
 | |
|     if (R_FAILED((rc))) {
 | |
|         tmemClose(&g_tmem);
 | |
|     }
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| Result amssuSetupUpdateWithVariation(void *buffer, size_t size, const char *path, bool exfat, u32 variation) {
 | |
|     Result rc = 0;
 | |
| 
 | |
|     if (buffer == NULL) {
 | |
|         rc = tmemCreate(&g_tmem, size, Perm_None);
 | |
|     } else {
 | |
|         rc = tmemCreateFromMemory(&g_tmem, buffer, size, Perm_None);
 | |
|     }
 | |
|     if (R_FAILED(rc)) return rc;
 | |
| 
 | |
|     char send_path[FS_MAX_PATH] = {0};
 | |
|     strncpy(send_path, path, FS_MAX_PATH-1);
 | |
|     send_path[FS_MAX_PATH-1] = 0;
 | |
| 
 | |
|     const struct {
 | |
|         u8 exfat;
 | |
|         u32 variation;
 | |
|         u64 size;
 | |
|     } in = { exfat, variation, g_tmem.size };
 | |
| 
 | |
|     rc = serviceDispatchIn(&g_amssuSrv, 3, in,
 | |
|         .in_num_handles = 1,
 | |
|         .in_handles = { g_tmem.handle },
 | |
|         .buffer_attrs = { SfBufferAttr_In | SfBufferAttr_HipcPointer | SfBufferAttr_FixedSize },
 | |
|         .buffers      = { { send_path, FS_MAX_PATH } },
 | |
|     );
 | |
|     if (R_FAILED((rc))) {
 | |
|         tmemClose(&g_tmem);
 | |
|     }
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| Result amssuRequestPrepareUpdate(AsyncResult *a) {
 | |
|     memset(a, 0, sizeof(*a));
 | |
| 
 | |
|     Handle event = INVALID_HANDLE;
 | |
|     Result rc = serviceDispatch(&g_amssuSrv, 4,
 | |
|         .out_num_objects = 1,
 | |
|         .out_objects = &a->s,
 | |
|         .out_handle_attrs = { SfOutHandleAttr_HipcCopy },
 | |
|         .out_handles = &event,
 | |
|     );
 | |
| 
 | |
|     if (R_SUCCEEDED(rc))
 | |
|         eventLoadRemote(&a->event, event, false);
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| Result amssuGetPrepareUpdateProgress(NsSystemUpdateProgress *out) {
 | |
|     return serviceDispatchOut(&g_amssuSrv, 5, *out);
 | |
| }
 | |
| 
 | |
| Result amssuHasPreparedUpdate(bool *out) {
 | |
|     u8 outval = 0;
 | |
|     Result rc = serviceDispatchOut(&g_amssuSrv, 6, outval);
 | |
|     if (R_SUCCEEDED(rc)) {
 | |
|         if (out) *out = outval & 1;
 | |
|     }
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| Result amssuApplyPreparedUpdate() {
 | |
|     return serviceDispatch(&g_amssuSrv, 7);
 | |
| } |