| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | // Copyright 2015 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  | #include <boost/optional.hpp>
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/common_paths.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | #include "common/file_util.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-15 19:09:05 +03:00
										 |  |  | #include "core/file_sys/file_backend.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | #include "core/hle/applets/applet.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-21 00:21:23 +09:00
										 |  |  | #include "core/hle/kernel/event.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/mutex.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							|  |  |  | #include "core/hle/kernel/shared_memory.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-15 19:09:05 +03:00
										 |  |  | #include "core/hle/romfs.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/hle/service/apt/apt.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | #include "core/hle/service/apt/apt_a.h"
 | 
					
						
							|  |  |  | #include "core/hle/service/apt/apt_s.h"
 | 
					
						
							|  |  |  | #include "core/hle/service/apt/apt_u.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-19 12:13:00 -05:00
										 |  |  | #include "core/hle/service/apt/bcfnt/bcfnt.h"
 | 
					
						
							| 
									
										
										
										
											2015-12-04 16:05:23 -05:00
										 |  |  | #include "core/hle/service/fs/archive.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | #include "core/hle/service/ptm/ptm.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include "core/hle/service/service.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-01 14:58:24 +02:00
										 |  |  | #include "core/hw/aes/ccm.h"
 | 
					
						
							|  |  |  | #include "core/hw/aes/key.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Service { | 
					
						
							|  |  |  | namespace APT { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Handle to shared memory region designated to for shared system font
 | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  | static Kernel::SharedPtr<Kernel::SharedMemory> shared_font_mem; | 
					
						
							| 
									
										
										
										
											2017-06-15 19:09:05 +03:00
										 |  |  | static bool shared_font_loaded = false; | 
					
						
							| 
									
										
										
										
											2016-04-17 21:12:37 -05:00
										 |  |  | static bool shared_font_relocated = false; | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  | static Kernel::SharedPtr<Kernel::Mutex> lock; | 
					
						
							| 
									
										
										
										
											2015-04-05 22:11:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  | static u32 cpu_percent; ///< CPU time available to the running application
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | // APT::CheckNew3DSApp will check this unknown_ns_state_field to determine processing mode
 | 
					
						
							|  |  |  | static u8 unknown_ns_state_field; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-14 18:26:43 +08:00
										 |  |  | static ScreencapPostPermission screen_capture_post_permission; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  | /// Parameter data to be returned in the next call to Glance/ReceiveParameter.
 | 
					
						
							|  |  |  | /// TODO(Subv): Use std::optional once we migrate to C++17.
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  | static boost::optional<MessageParameter> next_parameter; | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  | enum class AppletPos { Application = 0, Library = 1, System = 2, SysLibrary = 3, Resident = 4 }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static constexpr size_t NumAppletSlot = 4; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class AppletSlot : u8 { | 
					
						
							|  |  |  |     Application, | 
					
						
							|  |  |  |     SystemApplet, | 
					
						
							|  |  |  |     HomeMenu, | 
					
						
							|  |  |  |     LibraryApplet, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // An invalid tag
 | 
					
						
							|  |  |  |     Error, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-07 16:09:55 -05:00
										 |  |  | union AppletAttributes { | 
					
						
							|  |  |  |     u32 raw; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     BitField<0, 3, u32> applet_pos; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AppletAttributes() : raw(0) {} | 
					
						
							|  |  |  |     AppletAttributes(u32 attributes) : raw(attributes) {} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  | struct AppletSlotData { | 
					
						
							|  |  |  |     AppletId applet_id; | 
					
						
							|  |  |  |     AppletSlot slot; | 
					
						
							|  |  |  |     bool registered; | 
					
						
							| 
									
										
										
										
											2017-08-07 16:09:55 -05:00
										 |  |  |     AppletAttributes attributes; | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     Kernel::SharedPtr<Kernel::Event> notification_event; | 
					
						
							|  |  |  |     Kernel::SharedPtr<Kernel::Event> parameter_event; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Holds data about the concurrently running applets in the system.
 | 
					
						
							|  |  |  | static std::array<AppletSlotData, NumAppletSlot> applet_slots = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This overload returns nullptr if no applet with the specified id has been started.
 | 
					
						
							|  |  |  | static AppletSlotData* GetAppletSlotData(AppletId id) { | 
					
						
							|  |  |  |     auto GetSlot = [](AppletSlot slot) -> AppletSlotData* { | 
					
						
							|  |  |  |         return &applet_slots[static_cast<size_t>(slot)]; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (id == AppletId::Application) { | 
					
						
							|  |  |  |         auto* slot = GetSlot(AppletSlot::Application); | 
					
						
							|  |  |  |         if (slot->applet_id != AppletId::None) | 
					
						
							|  |  |  |             return slot; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (id == AppletId::AnySystemApplet) { | 
					
						
							|  |  |  |         auto* system_slot = GetSlot(AppletSlot::SystemApplet); | 
					
						
							|  |  |  |         if (system_slot->applet_id != AppletId::None) | 
					
						
							|  |  |  |             return system_slot; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // The Home Menu is also a system applet, but it lives in its own slot to be able to run
 | 
					
						
							|  |  |  |         // concurrently with other system applets.
 | 
					
						
							|  |  |  |         auto* home_slot = GetSlot(AppletSlot::HomeMenu); | 
					
						
							|  |  |  |         if (home_slot->applet_id != AppletId::None) | 
					
						
							|  |  |  |             return home_slot; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (id == AppletId::AnyLibraryApplet || id == AppletId::AnySysLibraryApplet) { | 
					
						
							|  |  |  |         auto* slot = GetSlot(AppletSlot::LibraryApplet); | 
					
						
							|  |  |  |         if (slot->applet_id == AppletId::None) | 
					
						
							|  |  |  |             return nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-07 16:09:55 -05:00
										 |  |  |         u32 applet_pos = slot->attributes.applet_pos; | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (id == AppletId::AnyLibraryApplet && applet_pos == static_cast<u32>(AppletPos::Library)) | 
					
						
							|  |  |  |             return slot; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (id == AppletId::AnySysLibraryApplet && | 
					
						
							|  |  |  |             applet_pos == static_cast<u32>(AppletPos::SysLibrary)) | 
					
						
							|  |  |  |             return slot; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (id == AppletId::HomeMenu || id == AppletId::AlternateMenu) { | 
					
						
							|  |  |  |         auto* slot = GetSlot(AppletSlot::HomeMenu); | 
					
						
							|  |  |  |         if (slot->applet_id != AppletId::None) | 
					
						
							|  |  |  |             return slot; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto& slot : applet_slots) { | 
					
						
							|  |  |  |         if (slot.applet_id == id) | 
					
						
							|  |  |  |             return &slot; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-07 16:09:55 -05:00
										 |  |  | static AppletSlotData* GetAppletSlotData(AppletAttributes attributes) { | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     // Mapping from AppletPos to AppletSlot
 | 
					
						
							|  |  |  |     static constexpr std::array<AppletSlot, 6> applet_position_slots = { | 
					
						
							|  |  |  |         AppletSlot::Application,   AppletSlot::LibraryApplet, AppletSlot::SystemApplet, | 
					
						
							|  |  |  |         AppletSlot::LibraryApplet, AppletSlot::Error,         AppletSlot::LibraryApplet}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-07 16:09:55 -05:00
										 |  |  |     u32 applet_pos = attributes.applet_pos; | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     if (applet_pos >= applet_position_slots.size()) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AppletSlot slot = applet_position_slots[applet_pos]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (slot == AppletSlot::Error) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return &applet_slots[static_cast<size_t>(slot)]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-27 15:21:06 -05:00
										 |  |  | void SendParameter(const MessageParameter& parameter) { | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  |     next_parameter = parameter; | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     // Signal the event to let the receiver know that a new parameter is ready to be read
 | 
					
						
							|  |  |  |     auto* const slot_data = GetAppletSlotData(static_cast<AppletId>(parameter.destination_id)); | 
					
						
							|  |  |  |     ASSERT(slot_data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     slot_data->parameter_event->Signal(); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | void Initialize(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x2, 2, 0); // 0x20080
 | 
					
						
							|  |  |  |     u32 app_id = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     u32 attributes = rp.Pop<u32>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_DEBUG(Service_APT, "called app_id=0x%08X, attributes=0x%08X", app_id, attributes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto* const slot_data = GetAppletSlotData(attributes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Note: The real NS service does not check if the attributes value is valid before accessing
 | 
					
						
							|  |  |  |     // the data in the array
 | 
					
						
							|  |  |  |     ASSERT_MSG(slot_data, "Invalid application attributes"); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     if (slot_data->registered) { | 
					
						
							|  |  |  |         IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |         rb.Push(ResultCode(ErrorDescription::AlreadyExists, ErrorModule::Applet, | 
					
						
							|  |  |  |                            ErrorSummary::InvalidState, ErrorLevel::Status)); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     slot_data->applet_id = static_cast<AppletId>(app_id); | 
					
						
							| 
									
										
										
										
											2017-08-07 16:09:55 -05:00
										 |  |  |     slot_data->attributes.raw = attributes; | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 3); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |     rb.PushCopyHandles(Kernel::g_handle_table.Create(slot_data->notification_event).Unwrap(), | 
					
						
							|  |  |  |                        Kernel::g_handle_table.Create(slot_data->parameter_event).Unwrap()); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GetSharedFont(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x44, 0, 0); // 0x00440000
 | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(2, 2); | 
					
						
							| 
									
										
										
										
											2017-08-01 20:00:40 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Log in telemetry if the game uses the shared font
 | 
					
						
							|  |  |  |     Core::Telemetry().AddField(Telemetry::FieldType::Session, "RequiresSharedFont", true); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-15 19:09:05 +03:00
										 |  |  |     if (!shared_font_loaded) { | 
					
						
							| 
									
										
										
										
											2016-07-28 21:47:31 +02:00
										 |  |  |         LOG_ERROR(Service_APT, "shared font file missing - go dump it from your 3ds"); | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |         rb.Push<u32>(-1); // TODO: Find the right error code
 | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |         rb.Skip(1 + 2, true); | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |         Core::System::GetInstance().SetStatus(Core::System::ResultStatus::ErrorSharedFont); | 
					
						
							| 
									
										
										
										
											2016-07-28 21:47:31 +02:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     // The shared font has to be relocated to the new address before being passed to the
 | 
					
						
							|  |  |  |     // application.
 | 
					
						
							|  |  |  |     VAddr target_address = | 
					
						
							| 
									
										
										
										
											2017-06-21 20:21:49 -07:00
										 |  |  |         Memory::PhysicalToVirtualAddress(shared_font_mem->linear_heap_phys_address).value(); | 
					
						
							| 
									
										
										
										
											2016-04-19 12:13:00 -05:00
										 |  |  |     if (!shared_font_relocated) { | 
					
						
							| 
									
										
										
										
											2016-08-27 01:04:26 -07:00
										 |  |  |         BCFNT::RelocateSharedFont(shared_font_mem, target_address); | 
					
						
							| 
									
										
										
										
											2016-04-19 12:13:00 -05:00
										 |  |  |         shared_font_relocated = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     // Since the SharedMemory interface doesn't provide the address at which the memory was
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  |     // allocated, the real APT service calculates this address by scanning the entire address space
 | 
					
						
							|  |  |  |     // (using svcQueryMemory) and searches for an allocation of the same size as the Shared Font.
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     rb.Push(target_address); | 
					
						
							| 
									
										
										
										
											2017-06-18 19:03:15 -07:00
										 |  |  |     rb.PushCopyHandles(Kernel::g_handle_table.Create(shared_font_mem).Unwrap()); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void NotifyToWait(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x43, 1, 0); // 0x430040
 | 
					
						
							|  |  |  |     u32 app_id = rp.Pop<u32>(); | 
					
						
							|  |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  |     LOG_WARNING(Service_APT, "(STUBBED) app_id=%u", app_id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GetLockHandle(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x1, 1, 0); // 0x10040
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-23 18:52:57 -05:00
										 |  |  |     // Bits [0:2] are the applet type (System, Library, etc)
 | 
					
						
							|  |  |  |     // Bit 5 tells the application that there's a pending APT parameter,
 | 
					
						
							|  |  |  |     // this will cause the app to wait until parameter_event is signaled.
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     u32 applet_attributes = rp.Pop<u32>(); | 
					
						
							|  |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(3, 2); | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // TODO(Subv): The output attributes should have an AppletPos of either Library or System |
 | 
					
						
							|  |  |  |     // Library (depending on the type of the last launched applet) if the input attributes'
 | 
					
						
							|  |  |  |     // AppletPos has the Library bit set.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     rb.Push(applet_attributes); // Applet Attributes, this value is passed to Enable.
 | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |     rb.Push<u32>(0);            // Least significant bit = power button state
 | 
					
						
							| 
									
										
										
										
											2017-06-18 19:03:15 -07:00
										 |  |  |     Kernel::Handle handle_copy = Kernel::g_handle_table.Create(lock).Unwrap(); | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     rb.PushCopyHandles(handle_copy); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     LOG_WARNING(Service_APT, "(STUBBED) called handle=0x%08X applet_attributes=0x%08X", handle_copy, | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 applet_attributes); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Enable(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x3, 1, 0); // 0x30040
 | 
					
						
							|  |  |  |     u32 attributes = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     LOG_DEBUG(Service_APT, "called attributes=0x%08X", attributes); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto* const slot_data = GetAppletSlotData(attributes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!slot_data) { | 
					
						
							|  |  |  |         rb.Push(ResultCode(ErrCodes::InvalidAppletSlot, ErrorModule::Applet, | 
					
						
							|  |  |  |                            ErrorSummary::InvalidState, ErrorLevel::Status)); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     slot_data->registered = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GetAppletManInfo(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x5, 1, 0); // 0x50040
 | 
					
						
							|  |  |  |     u32 unk = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2017-03-19 01:33:56 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(5, 0); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |     rb.Push<u32>(0); | 
					
						
							|  |  |  |     rb.Push<u32>(0); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     rb.Push(static_cast<u32>(AppletId::HomeMenu));    // Home menu AppID
 | 
					
						
							|  |  |  |     rb.Push(static_cast<u32>(AppletId::Application)); // TODO(purpasmart96): Do this correctly
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     LOG_WARNING(Service_APT, "(STUBBED) called unk=0x%08X", unk); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void IsRegistered(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x9, 1, 0); // 0x90040
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     AppletId app_id = static_cast<AppletId>(rp.Pop<u32>()); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2015-07-23 18:52:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     auto* const slot_data = GetAppletSlotData(app_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Check if an LLE applet was registered first, then fallback to HLE applets
 | 
					
						
							|  |  |  |     bool is_registered = slot_data && slot_data->registered; | 
					
						
							| 
									
										
										
										
											2015-07-23 18:52:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     if (!is_registered) { | 
					
						
							|  |  |  |         if (app_id == AppletId::AnyLibraryApplet) { | 
					
						
							|  |  |  |             is_registered = HLE::Applets::IsLibraryAppletRunning(); | 
					
						
							|  |  |  |         } else if (auto applet = HLE::Applets::Applet::Get(app_id)) { | 
					
						
							|  |  |  |             // The applet exists, set it as registered.
 | 
					
						
							|  |  |  |             is_registered = true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-07-23 18:52:57 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     rb.Push(is_registered); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     LOG_DEBUG(Service_APT, "called app_id=0x%08X", static_cast<u32>(app_id)); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void InquireNotification(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0xB, 1, 0); // 0xB0040
 | 
					
						
							|  |  |  |     u32 app_id = rp.Pop<u32>(); | 
					
						
							|  |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS);                     // No error
 | 
					
						
							|  |  |  |     rb.Push(static_cast<u32>(SignalType::None)); // Signal type
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  |     LOG_WARNING(Service_APT, "(STUBBED) called app_id=0x%08X", app_id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SendParameter(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0xC, 4, 4); // 0xC0104
 | 
					
						
							|  |  |  |     u32 src_app_id = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 dst_app_id = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 signal_type = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 buffer_size = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     Kernel::Handle handle = rp.PopHandle(); | 
					
						
							|  |  |  |     size_t size; | 
					
						
							|  |  |  |     VAddr buffer = rp.PopStaticBuffer(&size); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::shared_ptr<HLE::Applets::Applet> dest_applet = | 
					
						
							|  |  |  |         HLE::Applets::Applet::Get(static_cast<AppletId>(dst_app_id)); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 15:03:06 -05:00
										 |  |  |     LOG_DEBUG(Service_APT, | 
					
						
							|  |  |  |               "called src_app_id=0x%08X, dst_app_id=0x%08X, signal_type=0x%08X," | 
					
						
							|  |  |  |               "buffer_size=0x%08X, handle=0x%08X, size=0x%08zX, in_param_buffer_ptr=0x%08X", | 
					
						
							|  |  |  |               src_app_id, dst_app_id, signal_type, buffer_size, handle, size, buffer); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:19:55 -05:00
										 |  |  |     // A new parameter can not be sent if the previous one hasn't been consumed yet
 | 
					
						
							|  |  |  |     if (next_parameter) { | 
					
						
							|  |  |  |         rb.Push(ResultCode(ErrCodes::ParameterPresent, ErrorModule::Applet, | 
					
						
							|  |  |  |                            ErrorSummary::InvalidState, ErrorLevel::Status)); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  |     if (dest_applet == nullptr) { | 
					
						
							|  |  |  |         LOG_ERROR(Service_APT, "Unknown applet id=0x%08X", dst_app_id); | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |         rb.Push<u32>(-1); // TODO(Subv): Find the right error code
 | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-25 20:34:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  |     MessageParameter param; | 
					
						
							|  |  |  |     param.destination_id = dst_app_id; | 
					
						
							|  |  |  |     param.sender_id = src_app_id; | 
					
						
							|  |  |  |     param.object = Kernel::g_handle_table.GetGeneric(handle); | 
					
						
							|  |  |  |     param.signal = signal_type; | 
					
						
							| 
									
										
										
										
											2016-04-16 11:18:49 +01:00
										 |  |  |     param.buffer.resize(buffer_size); | 
					
						
							|  |  |  |     Memory::ReadBlock(buffer, param.buffer.data(), param.buffer.size()); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     rb.Push(dest_applet->ReceiveParameter(param)); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ReceiveParameter(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0xD, 2, 0); // 0xD0080
 | 
					
						
							|  |  |  |     u32 app_id = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 buffer_size = rp.Pop<u32>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     size_t static_buff_size; | 
					
						
							|  |  |  |     VAddr buffer = rp.PeekStaticBuffer(0, &static_buff_size); | 
					
						
							|  |  |  |     if (buffer_size > static_buff_size) | 
					
						
							| 
									
										
										
										
											2017-03-19 01:33:56 +01:00
										 |  |  |         LOG_WARNING( | 
					
						
							|  |  |  |             Service_APT, | 
					
						
							|  |  |  |             "buffer_size is bigger than the size in the buffer descriptor (0x%08X > 0x%08zX)", | 
					
						
							|  |  |  |             buffer_size, static_buff_size); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 15:03:06 -05:00
										 |  |  |     LOG_DEBUG(Service_APT, "called app_id=0x%08X, buffer_size=0x%08zX", app_id, buffer_size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:02:40 -05:00
										 |  |  |     if (!next_parameter) { | 
					
						
							|  |  |  |         IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |         rb.Push(ResultCode(ErrorDescription::NoData, ErrorModule::Applet, | 
					
						
							|  |  |  |                            ErrorSummary::InvalidState, ErrorLevel::Status)); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (next_parameter->destination_id != app_id) { | 
					
						
							|  |  |  |         IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |         rb.Push(ResultCode(ErrorDescription::NotFound, ErrorModule::Applet, ErrorSummary::NotFound, | 
					
						
							|  |  |  |                            ErrorLevel::Status)); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(4, 4); | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     rb.Push(next_parameter->sender_id); | 
					
						
							|  |  |  |     rb.Push(next_parameter->signal); // Signal type
 | 
					
						
							|  |  |  |     ASSERT_MSG(next_parameter->buffer.size() <= buffer_size, "Input static buffer is too small !"); | 
					
						
							|  |  |  |     rb.Push(static_cast<u32>(next_parameter->buffer.size())); // Parameter buffer size
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     rb.PushMoveHandles((next_parameter->object != nullptr) | 
					
						
							|  |  |  |                            ? Kernel::g_handle_table.Create(next_parameter->object).Unwrap() | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |                            : 0); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     rb.PushStaticBuffer(buffer, static_cast<u32>(next_parameter->buffer.size()), 0); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     Memory::WriteBlock(buffer, next_parameter->buffer.data(), next_parameter->buffer.size()); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:03:28 -05:00
										 |  |  |     // Clear the parameter
 | 
					
						
							|  |  |  |     next_parameter = boost::none; | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GlanceParameter(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0xE, 2, 0); // 0xE0080
 | 
					
						
							|  |  |  |     u32 app_id = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 buffer_size = rp.Pop<u32>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     size_t static_buff_size; | 
					
						
							|  |  |  |     VAddr buffer = rp.PeekStaticBuffer(0, &static_buff_size); | 
					
						
							|  |  |  |     if (buffer_size > static_buff_size) | 
					
						
							| 
									
										
										
										
											2017-03-19 01:33:56 +01:00
										 |  |  |         LOG_WARNING( | 
					
						
							|  |  |  |             Service_APT, | 
					
						
							|  |  |  |             "buffer_size is bigger than the size in the buffer descriptor (0x%08X > 0x%08zX)", | 
					
						
							|  |  |  |             buffer_size, static_buff_size); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 15:03:06 -05:00
										 |  |  |     LOG_DEBUG(Service_APT, "called app_id=0x%08X, buffer_size=0x%08zX", app_id, buffer_size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:02:40 -05:00
										 |  |  |     if (!next_parameter) { | 
					
						
							|  |  |  |         IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |         rb.Push(ResultCode(ErrorDescription::NoData, ErrorModule::Applet, | 
					
						
							|  |  |  |                            ErrorSummary::InvalidState, ErrorLevel::Status)); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (next_parameter->destination_id != app_id) { | 
					
						
							|  |  |  |         IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |         rb.Push(ResultCode(ErrorDescription::NotFound, ErrorModule::Applet, ErrorSummary::NotFound, | 
					
						
							|  |  |  |                            ErrorLevel::Status)); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(4, 4); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     rb.Push(next_parameter->sender_id); | 
					
						
							|  |  |  |     rb.Push(next_parameter->signal); // Signal type
 | 
					
						
							|  |  |  |     ASSERT_MSG(next_parameter->buffer.size() <= buffer_size, "Input static buffer is too small !"); | 
					
						
							|  |  |  |     rb.Push(static_cast<u32>(next_parameter->buffer.size())); // Parameter buffer size
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     rb.PushMoveHandles((next_parameter->object != nullptr) | 
					
						
							|  |  |  |                            ? Kernel::g_handle_table.Create(next_parameter->object).Unwrap() | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |                            : 0); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     rb.PushStaticBuffer(buffer, static_cast<u32>(next_parameter->buffer.size()), 0); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     Memory::WriteBlock(buffer, next_parameter->buffer.data(), next_parameter->buffer.size()); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:03:28 -05:00
										 |  |  |     // Note: The NS module always clears the DSPSleep and DSPWakeup signals even in GlanceParameter.
 | 
					
						
							|  |  |  |     if (next_parameter->signal == static_cast<u32>(SignalType::DspSleep) || | 
					
						
							|  |  |  |         next_parameter->signal == static_cast<u32>(SignalType::DspWakeup)) | 
					
						
							|  |  |  |         next_parameter = boost::none; | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CancelParameter(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0xF, 4, 0); // 0xF0100
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:09:45 -05:00
										 |  |  |     bool check_sender = rp.Pop<bool>(); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     u32 sender_appid = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2017-07-21 13:09:45 -05:00
										 |  |  |     bool check_receiver = rp.Pop<bool>(); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     u32 receiver_appid = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2017-07-21 13:09:45 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bool cancellation_success = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!next_parameter) { | 
					
						
							|  |  |  |         cancellation_success = false; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         if (check_sender && next_parameter->sender_id != sender_appid) | 
					
						
							|  |  |  |             cancellation_success = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (check_receiver && next_parameter->destination_id != receiver_appid) | 
					
						
							|  |  |  |             cancellation_success = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cancellation_success) | 
					
						
							|  |  |  |         next_parameter = boost::none; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); | 
					
						
							| 
									
										
										
										
											2017-07-21 13:09:45 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:09:45 -05:00
										 |  |  |     rb.Push(cancellation_success); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:09:45 -05:00
										 |  |  |     LOG_DEBUG(Service_APT, "called check_sender=%u, sender_appid=0x%08X, " | 
					
						
							|  |  |  |                            "check_receiver=%u, receiver_appid=0x%08X", | 
					
						
							|  |  |  |               check_sender, sender_appid, check_receiver, receiver_appid); | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void PrepareToStartApplication(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x15, 5, 0); // 0x00150140
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     u32 title_info1 = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 title_info2 = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 title_info3 = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 title_info4 = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 flags = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  |     if (flags & 0x00000100) { | 
					
						
							|  |  |  |         unknown_ns_state_field = 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     LOG_WARNING(Service_APT, | 
					
						
							|  |  |  |                 "(STUBBED) called title_info1=0x%08X, title_info2=0x%08X, title_info3=0x%08X," | 
					
						
							|  |  |  |                 "title_info4=0x%08X, flags=0x%08X", | 
					
						
							|  |  |  |                 title_info1, title_info2, title_info3, title_info4, flags); | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void StartApplication(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x1B, 3, 4); // 0x001B00C4
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     u32 buffer1_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 buffer2_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 flag = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     size_t size1; | 
					
						
							|  |  |  |     VAddr buffer1_ptr = rp.PopStaticBuffer(&size1); | 
					
						
							|  |  |  |     size_t size2; | 
					
						
							|  |  |  |     VAddr buffer2_ptr = rp.PopStaticBuffer(&size2); | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     LOG_WARNING(Service_APT, | 
					
						
							|  |  |  |                 "(STUBBED) called buffer1_size=0x%08X, buffer2_size=0x%08X, flag=0x%08X," | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |                 "size1=0x%08zX, buffer1_ptr=0x%08X, size2=0x%08zX, buffer2_ptr=0x%08X", | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 buffer1_size, buffer2_size, flag, size1, buffer1_ptr, size2, buffer2_ptr); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AppletUtility(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x4B, 3, 2); // 0x004B00C2
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // These are from 3dbrew - I'm not really sure what they're used for.
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     u32 utility_command = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 input_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 output_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     VAddr input_addr = rp.PopStaticBuffer(); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     VAddr output_addr = rp.PeekStaticBuffer(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     LOG_WARNING(Service_APT, | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |                 "(STUBBED) called command=0x%08X, input_size=0x%08X, output_size=0x%08X, " | 
					
						
							|  |  |  |                 "input_addr=0x%08X, output_addr=0x%08X", | 
					
						
							|  |  |  |                 utility_command, input_size, output_size, input_addr, output_addr); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SetAppCpuTimeLimit(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x4F, 2, 0); // 0x4F0080
 | 
					
						
							|  |  |  |     u32 value = rp.Pop<u32>(); | 
					
						
							|  |  |  |     cpu_percent = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (value != 1) { | 
					
						
							|  |  |  |         LOG_ERROR(Service_APT, "This value should be one, but is actually %u!", value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  |     LOG_WARNING(Service_APT, "(STUBBED) called cpu_percent=%u, value=%u", cpu_percent, value); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GetAppCpuTimeLimit(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x50, 1, 0); // 0x500040
 | 
					
						
							|  |  |  |     u32 value = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (value != 1) { | 
					
						
							|  |  |  |         LOG_ERROR(Service_APT, "This value should be one, but is actually %u!", value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							|  |  |  |     rb.Push(cpu_percent); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  |     LOG_WARNING(Service_APT, "(STUBBED) called value=%u", value); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | void PrepareToStartLibraryApplet(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x18, 1, 0); // 0x180040
 | 
					
						
							|  |  |  |     AppletId applet_id = static_cast<AppletId>(rp.Pop<u32>()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							| 
									
										
										
										
											2015-07-23 21:09:43 -05:00
										 |  |  |     auto applet = HLE::Applets::Applet::Get(applet_id); | 
					
						
							|  |  |  |     if (applet) { | 
					
						
							|  |  |  |         LOG_WARNING(Service_APT, "applet has already been started id=%08X", applet_id); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2015-07-23 21:09:43 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         rb.Push(HLE::Applets::Applet::Create(applet_id)); | 
					
						
							| 
									
										
										
										
											2015-07-23 21:09:43 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     LOG_DEBUG(Service_APT, "called applet_id=%08X", applet_id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void PreloadLibraryApplet(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x16, 1, 0); // 0x160040
 | 
					
						
							|  |  |  |     AppletId applet_id = static_cast<AppletId>(rp.Pop<u32>()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							| 
									
										
										
										
											2015-07-23 21:09:43 -05:00
										 |  |  |     auto applet = HLE::Applets::Applet::Get(applet_id); | 
					
						
							|  |  |  |     if (applet) { | 
					
						
							|  |  |  |         LOG_WARNING(Service_APT, "applet has already been started id=%08X", applet_id); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2015-07-23 21:09:43 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         rb.Push(HLE::Applets::Applet::Create(applet_id)); | 
					
						
							| 
									
										
										
										
											2015-07-23 21:09:43 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-26 11:00:26 -05:00
										 |  |  |     LOG_DEBUG(Service_APT, "called applet_id=%08X", applet_id); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void StartLibraryApplet(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x1E, 2, 4); // 0x1E0084
 | 
					
						
							|  |  |  |     AppletId applet_id = static_cast<AppletId>(rp.Pop<u32>()); | 
					
						
							| 
									
										
										
										
											2015-05-26 11:00:26 -05:00
										 |  |  |     std::shared_ptr<HLE::Applets::Applet> applet = HLE::Applets::Applet::Get(applet_id); | 
					
						
							| 
									
										
										
										
											2015-05-27 15:21:06 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-26 11:00:26 -05:00
										 |  |  |     LOG_DEBUG(Service_APT, "called applet_id=%08X", applet_id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  |     if (applet == nullptr) { | 
					
						
							| 
									
										
										
										
											2015-05-26 11:00:26 -05:00
										 |  |  |         LOG_ERROR(Service_APT, "unknown applet id=%08X", applet_id); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         IPC::RequestBuilder rb = rp.MakeBuilder(1, 0, false); | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |         rb.Push<u32>(-1); // TODO(Subv): Find the right error code
 | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     size_t buffer_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     Kernel::Handle handle = rp.PopHandle(); | 
					
						
							|  |  |  |     VAddr buffer_addr = rp.PopStaticBuffer(); | 
					
						
							| 
									
										
										
										
											2016-04-16 11:18:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  |     AppletStartupParameter parameter; | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     parameter.object = Kernel::g_handle_table.GetGeneric(handle); | 
					
						
							| 
									
										
										
										
											2016-04-16 11:18:49 +01:00
										 |  |  |     parameter.buffer.resize(buffer_size); | 
					
						
							|  |  |  |     Memory::ReadBlock(buffer_addr, parameter.buffer.data(), parameter.buffer.size()); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |     rb.Push(applet->Start(parameter)); | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-23 19:57:56 -08:00
										 |  |  | void CancelLibraryApplet(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x3B, 1, 0); // 0x003B0040
 | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |     bool exiting = rp.Pop<bool>(); | 
					
						
							| 
									
										
										
										
											2016-11-23 19:57:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							| 
									
										
										
										
											2017-03-19 01:33:56 +01:00
										 |  |  |     rb.Push<u32>(1); // TODO: Find the return code meaning
 | 
					
						
							| 
									
										
										
										
											2016-11-23 19:57:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     LOG_WARNING(Service_APT, "(STUBBED) called exiting=%d", exiting); | 
					
						
							| 
									
										
										
										
											2016-11-23 19:57:56 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-14 18:26:43 +08:00
										 |  |  | void SetScreenCapPostPermission(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x55, 1, 0); // 0x00550040
 | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     screen_capture_post_permission = static_cast<ScreencapPostPermission>(rp.Pop<u32>() & 0xF); | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     LOG_WARNING(Service_APT, "(STUBBED) screen_capture_post_permission=%u", | 
					
						
							|  |  |  |                 screen_capture_post_permission); | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-14 18:26:43 +08:00
										 |  |  | void GetScreenCapPostPermission(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x56, 0, 0); // 0x00560000
 | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); // No error
 | 
					
						
							|  |  |  |     rb.Push(static_cast<u32>(screen_capture_post_permission)); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     LOG_WARNING(Service_APT, "(STUBBED) screen_capture_post_permission=%u", | 
					
						
							|  |  |  |                 screen_capture_post_permission); | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 16:05:23 -05:00
										 |  |  | void GetAppletInfo(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x6, 1, 0); // 0x60040
 | 
					
						
							|  |  |  |     auto app_id = static_cast<AppletId>(rp.Pop<u32>()); | 
					
						
							| 
									
										
										
										
											2015-12-04 16:05:23 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (auto applet = HLE::Applets::Applet::Get(app_id)) { | 
					
						
							|  |  |  |         // TODO(Subv): Get the title id for the current applet and write it in the response[2-3]
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         IPC::RequestBuilder rb = rp.MakeBuilder(7, 0); | 
					
						
							|  |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2017-03-19 01:33:56 +01:00
										 |  |  |         u64 title_id = 0; | 
					
						
							|  |  |  |         rb.Push(title_id); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         rb.Push(static_cast<u32>(Service::FS::MediaType::NAND)); | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |         rb.Push(true);   // Registered
 | 
					
						
							|  |  |  |         rb.Push(true);   // Loaded
 | 
					
						
							|  |  |  |         rb.Push<u32>(0); // Applet Attributes
 | 
					
						
							| 
									
										
										
										
											2015-12-04 16:05:23 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); | 
					
						
							|  |  |  |         rb.Push(ResultCode(ErrorDescription::NotFound, ErrorModule::Applet, ErrorSummary::NotFound, | 
					
						
							|  |  |  |                            ErrorLevel::Status)); | 
					
						
							| 
									
										
										
										
											2015-12-04 16:05:23 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     LOG_WARNING(Service_APT, "(stubbed) called appid=%u", app_id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-29 05:25:05 +08:00
										 |  |  | void GetStartupArgument(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x51, 2, 0); // 0x00510080
 | 
					
						
							|  |  |  |     u32 parameter_size = rp.Pop<u32>(); | 
					
						
							| 
									
										
										
										
											2017-03-19 01:33:56 +01:00
										 |  |  |     StartupArgumentType startup_argument_type = static_cast<StartupArgumentType>(rp.Pop<u8>()); | 
					
						
							| 
									
										
										
										
											2016-03-29 05:25:05 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parameter_size >= 0x300) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         LOG_ERROR( | 
					
						
							|  |  |  |             Service_APT, | 
					
						
							|  |  |  |             "Parameter size is outside the valid range (capped to 0x300): parameter_size=0x%08x", | 
					
						
							|  |  |  |             parameter_size); | 
					
						
							| 
									
										
										
										
											2016-03-29 05:25:05 +08:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     size_t static_buff_size; | 
					
						
							|  |  |  |     VAddr addr = rp.PeekStaticBuffer(0, &static_buff_size); | 
					
						
							|  |  |  |     if (parameter_size > static_buff_size) | 
					
						
							| 
									
										
										
										
											2017-03-19 01:33:56 +01:00
										 |  |  |         LOG_WARNING( | 
					
						
							|  |  |  |             Service_APT, | 
					
						
							|  |  |  |             "parameter_size is bigger than the size in the buffer descriptor (0x%08X > 0x%08zX)", | 
					
						
							|  |  |  |             parameter_size, static_buff_size); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-02 21:48:10 +03:00
										 |  |  |     if (addr && parameter_size) { | 
					
						
							|  |  |  |         Memory::ZeroBlock(addr, parameter_size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_WARNING(Service_APT, "(stubbed) called startup_argument_type=%u , parameter_size=0x%08x", | 
					
						
							|  |  |  |                 startup_argument_type, parameter_size); | 
					
						
							| 
									
										
										
										
											2016-03-29 05:25:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(2, 2); | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |     rb.Push<u32>(0); | 
					
						
							| 
									
										
										
										
											2017-03-18 11:47:40 +01:00
										 |  |  |     rb.PushStaticBuffer(addr, parameter_size, 0); | 
					
						
							| 
									
										
										
										
											2016-03-29 05:25:05 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-01 14:58:24 +02:00
										 |  |  | void Wrap(Service::Interface* self) { | 
					
						
							|  |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x46, 4, 4); | 
					
						
							|  |  |  |     const u32 output_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     const u32 input_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     const u32 nonce_offset = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 nonce_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     size_t desc_size; | 
					
						
							|  |  |  |     IPC::MappedBufferPermissions desc_permission; | 
					
						
							|  |  |  |     const VAddr input = rp.PopMappedBuffer(&desc_size, &desc_permission); | 
					
						
							|  |  |  |     ASSERT(desc_size == input_size && desc_permission == IPC::MappedBufferPermissions::R); | 
					
						
							|  |  |  |     const VAddr output = rp.PopMappedBuffer(&desc_size, &desc_permission); | 
					
						
							|  |  |  |     ASSERT(desc_size == output_size && desc_permission == IPC::MappedBufferPermissions::W); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Note: real 3DS still returns SUCCESS when the sizes don't match. It seems that it doesn't
 | 
					
						
							|  |  |  |     // check the buffer size and writes data with potential overflow.
 | 
					
						
							|  |  |  |     ASSERT_MSG(output_size == input_size + HW::AES::CCM_MAC_SIZE, | 
					
						
							|  |  |  |                "input_size (%d) doesn't match to output_size (%d)", input_size, output_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_DEBUG(Service_APT, "called, output_size=%u, input_size=%u, nonce_offset=%u, nonce_size=%u", | 
					
						
							|  |  |  |               output_size, input_size, nonce_offset, nonce_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Note: This weird nonce size modification is verified against real 3DS
 | 
					
						
							|  |  |  |     nonce_size = std::min<u32>(nonce_size & ~3, HW::AES::CCM_NONCE_SIZE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Reads nonce and concatenates the rest of the input as plaintext
 | 
					
						
							|  |  |  |     HW::AES::CCMNonce nonce{}; | 
					
						
							|  |  |  |     Memory::ReadBlock(input + nonce_offset, nonce.data(), nonce_size); | 
					
						
							|  |  |  |     u32 pdata_size = input_size - nonce_size; | 
					
						
							|  |  |  |     std::vector<u8> pdata(pdata_size); | 
					
						
							|  |  |  |     Memory::ReadBlock(input, pdata.data(), nonce_offset); | 
					
						
							|  |  |  |     Memory::ReadBlock(input + nonce_offset + nonce_size, pdata.data() + nonce_offset, | 
					
						
							|  |  |  |                       pdata_size - nonce_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Encrypts the plaintext using AES-CCM
 | 
					
						
							|  |  |  |     auto cipher = HW::AES::EncryptSignCCM(pdata, nonce, HW::AES::KeySlotID::APTWrap); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Puts the nonce to the beginning of the output, with ciphertext followed
 | 
					
						
							|  |  |  |     Memory::WriteBlock(output, nonce.data(), nonce_size); | 
					
						
							|  |  |  |     Memory::WriteBlock(output + nonce_size, cipher.data(), cipher.size()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 4); | 
					
						
							|  |  |  |     rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Unmap buffer
 | 
					
						
							|  |  |  |     rb.PushMappedBuffer(input, input_size, IPC::MappedBufferPermissions::R); | 
					
						
							|  |  |  |     rb.PushMappedBuffer(output, output_size, IPC::MappedBufferPermissions::W); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Unwrap(Service::Interface* self) { | 
					
						
							|  |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x47, 4, 4); | 
					
						
							|  |  |  |     const u32 output_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     const u32 input_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     const u32 nonce_offset = rp.Pop<u32>(); | 
					
						
							|  |  |  |     u32 nonce_size = rp.Pop<u32>(); | 
					
						
							|  |  |  |     size_t desc_size; | 
					
						
							|  |  |  |     IPC::MappedBufferPermissions desc_permission; | 
					
						
							|  |  |  |     const VAddr input = rp.PopMappedBuffer(&desc_size, &desc_permission); | 
					
						
							|  |  |  |     ASSERT(desc_size == input_size && desc_permission == IPC::MappedBufferPermissions::R); | 
					
						
							|  |  |  |     const VAddr output = rp.PopMappedBuffer(&desc_size, &desc_permission); | 
					
						
							|  |  |  |     ASSERT(desc_size == output_size && desc_permission == IPC::MappedBufferPermissions::W); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Note: real 3DS still returns SUCCESS when the sizes don't match. It seems that it doesn't
 | 
					
						
							|  |  |  |     // check the buffer size and writes data with potential overflow.
 | 
					
						
							|  |  |  |     ASSERT_MSG(output_size == input_size - HW::AES::CCM_MAC_SIZE, | 
					
						
							|  |  |  |                "input_size (%d) doesn't match to output_size (%d)", input_size, output_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_DEBUG(Service_APT, "called, output_size=%u, input_size=%u, nonce_offset=%u, nonce_size=%u", | 
					
						
							|  |  |  |               output_size, input_size, nonce_offset, nonce_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Note: This weird nonce size modification is verified against real 3DS
 | 
					
						
							|  |  |  |     nonce_size = std::min<u32>(nonce_size & ~3, HW::AES::CCM_NONCE_SIZE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Reads nonce and cipher text
 | 
					
						
							|  |  |  |     HW::AES::CCMNonce nonce{}; | 
					
						
							|  |  |  |     Memory::ReadBlock(input, nonce.data(), nonce_size); | 
					
						
							|  |  |  |     u32 cipher_size = input_size - nonce_size; | 
					
						
							|  |  |  |     std::vector<u8> cipher(cipher_size); | 
					
						
							|  |  |  |     Memory::ReadBlock(input + nonce_size, cipher.data(), cipher_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Decrypts the ciphertext using AES-CCM
 | 
					
						
							|  |  |  |     auto pdata = HW::AES::DecryptVerifyCCM(cipher, nonce, HW::AES::KeySlotID::APTWrap); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 11:45:19 +02:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(1, 4); | 
					
						
							| 
									
										
										
										
											2017-01-01 14:58:24 +02:00
										 |  |  |     if (!pdata.empty()) { | 
					
						
							|  |  |  |         // Splits the plaintext and put the nonce in between
 | 
					
						
							|  |  |  |         Memory::WriteBlock(output, pdata.data(), nonce_offset); | 
					
						
							|  |  |  |         Memory::WriteBlock(output + nonce_offset, nonce.data(), nonce_size); | 
					
						
							|  |  |  |         Memory::WriteBlock(output + nonce_offset + nonce_size, pdata.data() + nonce_offset, | 
					
						
							|  |  |  |                            pdata.size() - nonce_offset); | 
					
						
							|  |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         LOG_ERROR(Service_APT, "Failed to decrypt data"); | 
					
						
							|  |  |  |         rb.Push(ResultCode(static_cast<ErrorDescription>(1), ErrorModule::PS, | 
					
						
							|  |  |  |                            ErrorSummary::WrongArgument, ErrorLevel::Status)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Unmap buffer
 | 
					
						
							|  |  |  |     rb.PushMappedBuffer(input, input_size, IPC::MappedBufferPermissions::R); | 
					
						
							|  |  |  |     rb.PushMappedBuffer(output, output_size, IPC::MappedBufferPermissions::W); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | void CheckNew3DSApp(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x101, 0, 0); // 0x01010000
 | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  |     if (unknown_ns_state_field) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         rb.Push(RESULT_SUCCESS); | 
					
						
							| 
									
										
										
										
											2017-02-11 15:07:12 +01:00
										 |  |  |         rb.Push<u32>(0); | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |         PTM::CheckNew3DS(rb); | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LOG_WARNING(Service_APT, "(STUBBED) called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckNew3DS(Service::Interface* self) { | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     IPC::RequestParser rp(Kernel::GetCommandBuffer(), 0x102, 0, 0); // 0x01020000
 | 
					
						
							|  |  |  |     IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-28 17:11:14 +01:00
										 |  |  |     PTM::CheckNew3DS(rb); | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     LOG_WARNING(Service_APT, "(STUBBED) called"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-15 19:09:05 +03:00
										 |  |  | static u32 DecompressLZ11(const u8* in, u8* out) { | 
					
						
							|  |  |  |     u32_le decompressed_size; | 
					
						
							|  |  |  |     memcpy(&decompressed_size, in, sizeof(u32)); | 
					
						
							|  |  |  |     in += 4; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u8 type = decompressed_size & 0xFF; | 
					
						
							|  |  |  |     ASSERT(type == 0x11); | 
					
						
							|  |  |  |     decompressed_size >>= 8; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 current_out_size = 0; | 
					
						
							|  |  |  |     u8 flags = 0, mask = 1; | 
					
						
							|  |  |  |     while (current_out_size < decompressed_size) { | 
					
						
							|  |  |  |         if (mask == 1) { | 
					
						
							|  |  |  |             flags = *(in++); | 
					
						
							|  |  |  |             mask = 0x80; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             mask >>= 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (flags & mask) { | 
					
						
							|  |  |  |             u8 byte1 = *(in++); | 
					
						
							|  |  |  |             u32 length = byte1 >> 4; | 
					
						
							|  |  |  |             u32 offset; | 
					
						
							|  |  |  |             if (length == 0) { | 
					
						
							|  |  |  |                 u8 byte2 = *(in++); | 
					
						
							|  |  |  |                 u8 byte3 = *(in++); | 
					
						
							|  |  |  |                 length = (((byte1 & 0x0F) << 4) | (byte2 >> 4)) + 0x11; | 
					
						
							|  |  |  |                 offset = (((byte2 & 0x0F) << 8) | byte3) + 0x1; | 
					
						
							|  |  |  |             } else if (length == 1) { | 
					
						
							|  |  |  |                 u8 byte2 = *(in++); | 
					
						
							|  |  |  |                 u8 byte3 = *(in++); | 
					
						
							|  |  |  |                 u8 byte4 = *(in++); | 
					
						
							|  |  |  |                 length = (((byte1 & 0x0F) << 12) | (byte2 << 4) | (byte3 >> 4)) + 0x111; | 
					
						
							|  |  |  |                 offset = (((byte3 & 0x0F) << 8) | byte4) + 0x1; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 u8 byte2 = *(in++); | 
					
						
							|  |  |  |                 length = (byte1 >> 4) + 0x1; | 
					
						
							|  |  |  |                 offset = (((byte1 & 0x0F) << 8) | byte2) + 0x1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (u32 i = 0; i < length; i++) { | 
					
						
							|  |  |  |                 *out = *(out - offset); | 
					
						
							|  |  |  |                 ++out; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             current_out_size += length; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             *(out++) = *(in++); | 
					
						
							|  |  |  |             current_out_size++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return decompressed_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool LoadSharedFont() { | 
					
						
							|  |  |  |     // TODO (wwylele): load different font archive for region CHN/KOR/TWN
 | 
					
						
							|  |  |  |     const u64_le shared_font_archive_id_low = 0x0004009b00014002; | 
					
						
							|  |  |  |     const u64_le shared_font_archive_id_high = 0x00000001ffffff00; | 
					
						
							|  |  |  |     std::vector<u8> shared_font_archive_id(16); | 
					
						
							|  |  |  |     std::memcpy(&shared_font_archive_id[0], &shared_font_archive_id_low, sizeof(u64)); | 
					
						
							|  |  |  |     std::memcpy(&shared_font_archive_id[8], &shared_font_archive_id_high, sizeof(u64)); | 
					
						
							|  |  |  |     FileSys::Path archive_path(shared_font_archive_id); | 
					
						
							|  |  |  |     auto archive_result = Service::FS::OpenArchive(Service::FS::ArchiveIdCode::NCCH, archive_path); | 
					
						
							|  |  |  |     if (archive_result.Failed()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::vector<u8> romfs_path(20, 0); // 20-byte all zero path for opening RomFS
 | 
					
						
							|  |  |  |     FileSys::Path file_path(romfs_path); | 
					
						
							|  |  |  |     FileSys::Mode open_mode = {}; | 
					
						
							|  |  |  |     open_mode.read_flag.Assign(1); | 
					
						
							|  |  |  |     auto file_result = Service::FS::OpenFileFromArchive(*archive_result, file_path, open_mode); | 
					
						
							|  |  |  |     if (file_result.Failed()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto romfs = std::move(file_result).Unwrap(); | 
					
						
							|  |  |  |     std::vector<u8> romfs_buffer(romfs->backend->GetSize()); | 
					
						
							|  |  |  |     romfs->backend->Read(0, romfs_buffer.size(), romfs_buffer.data()); | 
					
						
							|  |  |  |     romfs->backend->Close(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const u8* font_file = RomFS::GetFilePointer(romfs_buffer.data(), {u"cbf_std.bcfnt.lz"}); | 
					
						
							|  |  |  |     if (font_file == nullptr) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct { | 
					
						
							|  |  |  |         u32_le status; | 
					
						
							|  |  |  |         u32_le region; | 
					
						
							|  |  |  |         u32_le decompressed_size; | 
					
						
							|  |  |  |         INSERT_PADDING_WORDS(0x1D); | 
					
						
							|  |  |  |     } shared_font_header{}; | 
					
						
							|  |  |  |     static_assert(sizeof(shared_font_header) == 0x80, "shared_font_header has incorrect size"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     shared_font_header.status = 2; // successfully loaded
 | 
					
						
							|  |  |  |     shared_font_header.region = 1; // region JPN/EUR/USA
 | 
					
						
							|  |  |  |     shared_font_header.decompressed_size = | 
					
						
							|  |  |  |         DecompressLZ11(font_file, shared_font_mem->GetPointer(0x80)); | 
					
						
							|  |  |  |     std::memcpy(shared_font_mem->GetPointer(), &shared_font_header, sizeof(shared_font_header)); | 
					
						
							|  |  |  |     *shared_font_mem->GetPointer(0x83) = 'U'; // Change the magic from "CFNT" to "CFNU"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool LoadLegacySharedFont() { | 
					
						
							|  |  |  |     // This is the legacy method to load shared font.
 | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  |     // The expected format is a decrypted, uncompressed BCFNT file with the 0x80 byte header
 | 
					
						
							|  |  |  |     // generated by the APT:U service. The best way to get is by dumping it from RAM. We've provided
 | 
					
						
							|  |  |  |     // a homebrew app to do this: https://github.com/citra-emu/3dsutils. Put the resulting file
 | 
					
						
							|  |  |  |     // "shared_font.bin" in the Citra "sysdata" directory.
 | 
					
						
							|  |  |  |     std::string filepath = FileUtil::GetUserPath(D_SYSDATA_IDX) + SHARED_FONT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FileUtil::CreateFullPath(filepath); // Create path if not already created
 | 
					
						
							|  |  |  |     FileUtil::IOFile file(filepath, "rb"); | 
					
						
							|  |  |  |     if (file.IsOpen()) { | 
					
						
							| 
									
										
										
										
											2016-04-17 21:07:52 -05:00
										 |  |  |         file.ReadBytes(shared_font_mem->GetPointer(), file.GetSize()); | 
					
						
							| 
									
										
										
										
											2017-06-15 19:09:05 +03:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Init() { | 
					
						
							|  |  |  |     AddService(new APT_A_Interface); | 
					
						
							|  |  |  |     AddService(new APT_S_Interface); | 
					
						
							|  |  |  |     AddService(new APT_U_Interface); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     HLE::Applets::Init(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     using Kernel::MemoryPermission; | 
					
						
							|  |  |  |     shared_font_mem = | 
					
						
							|  |  |  |         Kernel::SharedMemory::Create(nullptr, 0x332000, // 3272 KB
 | 
					
						
							|  |  |  |                                      MemoryPermission::ReadWrite, MemoryPermission::Read, 0, | 
					
						
							|  |  |  |                                      Kernel::MemoryRegion::SYSTEM, "APT:SharedFont"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LoadSharedFont()) { | 
					
						
							|  |  |  |         shared_font_loaded = true; | 
					
						
							|  |  |  |     } else if (LoadLegacySharedFont()) { | 
					
						
							|  |  |  |         LOG_WARNING(Service_APT, "Loaded shared font by legacy method"); | 
					
						
							|  |  |  |         shared_font_loaded = true; | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2017-06-15 19:09:05 +03:00
										 |  |  |         LOG_WARNING(Service_APT, "Unable to load shared font"); | 
					
						
							|  |  |  |         shared_font_loaded = false; | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     lock = Kernel::Mutex::Create(false, "APT_U:Lock"); | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  |     cpu_percent = 0; | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:05 +08:00
										 |  |  |     unknown_ns_state_field = 0; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     screen_capture_post_permission = | 
					
						
							|  |  |  |         ScreencapPostPermission::CleanThePermission; // TODO(JamePeng): verify the initial value
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     for (size_t slot = 0; slot < applet_slots.size(); ++slot) { | 
					
						
							|  |  |  |         auto& slot_data = applet_slots[slot]; | 
					
						
							|  |  |  |         slot_data.slot = static_cast<AppletSlot>(slot); | 
					
						
							|  |  |  |         slot_data.applet_id = AppletId::None; | 
					
						
							| 
									
										
										
										
											2017-08-07 16:09:55 -05:00
										 |  |  |         slot_data.attributes.raw = 0; | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |         slot_data.registered = false; | 
					
						
							|  |  |  |         slot_data.notification_event = | 
					
						
							|  |  |  |             Kernel::Event::Create(Kernel::ResetType::OneShot, "APT:Notification"); | 
					
						
							|  |  |  |         slot_data.parameter_event = | 
					
						
							|  |  |  |             Kernel::Event::Create(Kernel::ResetType::OneShot, "APT:Parameter"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-25 23:30:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     // Initialize the parameter to wake up the application.
 | 
					
						
							|  |  |  |     next_parameter.emplace(); | 
					
						
							|  |  |  |     next_parameter->signal = static_cast<u32>(SignalType::Wakeup); | 
					
						
							|  |  |  |     next_parameter->destination_id = static_cast<u32>(AppletId::Application); | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  |     applet_slots[static_cast<size_t>(AppletSlot::Application)].parameter_event->Signal(); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:54:16 -08:00
										 |  |  | void Shutdown() { | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  |     shared_font_mem = nullptr; | 
					
						
							| 
									
										
										
										
											2017-06-15 19:09:05 +03:00
										 |  |  |     shared_font_loaded = false; | 
					
						
							| 
									
										
										
										
											2016-04-17 21:12:37 -05:00
										 |  |  |     shared_font_relocated = false; | 
					
						
							| 
									
										
										
										
											2015-04-27 22:01:48 -04:00
										 |  |  |     lock = nullptr; | 
					
						
							| 
									
										
										
										
											2017-07-22 12:33:03 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (auto& slot : applet_slots) { | 
					
						
							|  |  |  |         slot.registered = false; | 
					
						
							|  |  |  |         slot.notification_event = nullptr; | 
					
						
							|  |  |  |         slot.parameter_event = nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-07-17 02:24:13 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:01:21 -05:00
										 |  |  |     next_parameter = boost::none; | 
					
						
							| 
									
										
										
										
											2015-07-17 02:24:13 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-26 11:00:26 -05:00
										 |  |  |     HLE::Applets::Shutdown(); | 
					
						
							| 
									
										
										
										
											2015-02-26 21:13:08 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace APT
 | 
					
						
							|  |  |  | } // namespace Service
 |