ServiceNvDrv.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  1. using ChocolArm64.Memory;
  2. using Ryujinx.Core.OsHle.Handles;
  3. using Ryujinx.Core.OsHle.Ipc;
  4. using Ryujinx.Core.OsHle.Utilities;
  5. using Ryujinx.Graphics.Gpu;
  6. using System;
  7. using System.Collections.Generic;
  8. namespace Ryujinx.Core.OsHle.Services
  9. {
  10. static partial class Service
  11. {
  12. private delegate long ServiceProcessRequest(ServiceCtx Context);
  13. private static Dictionary<(string, int), ServiceProcessRequest> IoctlCmds =
  14. new Dictionary<(string, int), ServiceProcessRequest>()
  15. {
  16. { ("/dev/nvhost-as-gpu", 0x4101), NvGpuAsIoctlBindChannel },
  17. { ("/dev/nvhost-as-gpu", 0x4102), NvGpuAsIoctlAllocSpace },
  18. { ("/dev/nvhost-as-gpu", 0x4106), NvGpuAsIoctlMapBufferEx },
  19. { ("/dev/nvhost-as-gpu", 0x4108), NvGpuAsIoctlGetVaRegions },
  20. { ("/dev/nvhost-as-gpu", 0x4109), NvGpuAsIoctlInitializeEx },
  21. { ("/dev/nvhost-ctrl", 0x001b), NvHostIoctlCtrlGetConfig },
  22. { ("/dev/nvhost-ctrl", 0x001d), NvHostIoctlCtrlEventWait },
  23. { ("/dev/nvhost-ctrl-gpu", 0x4701), NvGpuIoctlZcullGetCtxSize },
  24. { ("/dev/nvhost-ctrl-gpu", 0x4702), NvGpuIoctlZcullGetInfo },
  25. { ("/dev/nvhost-ctrl-gpu", 0x4705), NvGpuIoctlGetCharacteristics },
  26. { ("/dev/nvhost-ctrl-gpu", 0x4706), NvGpuIoctlGetTpcMasks },
  27. { ("/dev/nvhost-ctrl-gpu", 0x4714), NvGpuIoctlZbcGetActiveSlotMask },
  28. { ("/dev/nvhost-gpu", 0x4714), NvMapIoctlChannelSetUserData },
  29. { ("/dev/nvhost-gpu", 0x4801), NvMapIoctlChannelSetNvMap },
  30. { ("/dev/nvhost-gpu", 0x4808), NvMapIoctlChannelSubmitGpFifo },
  31. { ("/dev/nvhost-gpu", 0x4809), NvMapIoctlChannelAllocObjCtx },
  32. { ("/dev/nvhost-gpu", 0x480b), NvMapIoctlChannelZcullBind },
  33. { ("/dev/nvhost-gpu", 0x480c), NvMapIoctlChannelSetErrorNotifier },
  34. { ("/dev/nvhost-gpu", 0x480d), NvMapIoctlChannelSetPriority },
  35. { ("/dev/nvhost-gpu", 0x481a), NvMapIoctlChannelAllocGpFifoEx2 },
  36. { ("/dev/nvmap", 0x0101), NvMapIocCreate },
  37. { ("/dev/nvmap", 0x0103), NvMapIocFromId },
  38. { ("/dev/nvmap", 0x0104), NvMapIocAlloc },
  39. { ("/dev/nvmap", 0x0109), NvMapIocParam },
  40. { ("/dev/nvmap", 0x010e), NvMapIocGetId },
  41. };
  42. public static long NvDrvOpen(ServiceCtx Context)
  43. {
  44. long NamePtr = Context.Request.SendBuff[0].Position;
  45. string Name = AMemoryHelper.ReadAsciiString(Context.Memory, NamePtr);
  46. int Fd = Context.Ns.Os.Fds.GenerateId(new FileDesc(Name));
  47. Context.ResponseData.Write(Fd);
  48. Context.ResponseData.Write(0);
  49. return 0;
  50. }
  51. public static long NvDrvIoctl(ServiceCtx Context)
  52. {
  53. int Fd = Context.RequestData.ReadInt32();
  54. int Cmd = Context.RequestData.ReadInt32() & 0xffff;
  55. FileDesc FdData = Context.Ns.Os.Fds.GetData<FileDesc>(Fd);
  56. long Position = Context.Request.PtrBuff[0].Position;
  57. Context.ResponseData.Write(0);
  58. if (IoctlCmds.TryGetValue((FdData.Name, Cmd), out ServiceProcessRequest ProcReq))
  59. {
  60. return ProcReq(Context);
  61. }
  62. else
  63. {
  64. throw new NotImplementedException($"{FdData.Name} {Cmd:x4}");
  65. }
  66. }
  67. public static long NvDrvClose(ServiceCtx Context)
  68. {
  69. int Fd = Context.RequestData.ReadInt32();
  70. Context.Ns.Os.Fds.Delete(Fd);
  71. Context.ResponseData.Write(0);
  72. return 0;
  73. }
  74. public static long NvDrvInitialize(ServiceCtx Context)
  75. {
  76. long TransferMemSize = Context.RequestData.ReadInt64();
  77. int TransferMemHandle = Context.Request.HandleDesc.ToCopy[0];
  78. Context.ResponseData.Write(0);
  79. return 0;
  80. }
  81. public static long NvDrvQueryEvent(ServiceCtx Context)
  82. {
  83. int Fd = Context.RequestData.ReadInt32();
  84. int EventId = Context.RequestData.ReadInt32();
  85. Context.Response.HandleDesc = IpcHandleDesc.MakeCopy(0xcafe);
  86. Context.ResponseData.Write(0);
  87. return 0;
  88. }
  89. public static long NvDrvSetClientPid(ServiceCtx Context)
  90. {
  91. long Pid = Context.RequestData.ReadInt64();
  92. Context.ResponseData.Write(0);
  93. return 0;
  94. }
  95. private static long NvGpuAsIoctlBindChannel(ServiceCtx Context)
  96. {
  97. long Position = Context.Request.PtrBuff[0].Position;
  98. int Fd = Context.Memory.ReadInt32(Position);
  99. return 0;
  100. }
  101. private static long NvGpuAsIoctlAllocSpace(ServiceCtx Context)
  102. {
  103. long Position = Context.Request.PtrBuff[0].Position;
  104. MemReader Reader = new MemReader(Context.Memory, Position);
  105. int Pages = Reader.ReadInt32();
  106. int PageSize = Reader.ReadInt32();
  107. int Flags = Reader.ReadInt32();
  108. int Padding = Reader.ReadInt32();
  109. long Align = Reader.ReadInt64();
  110. if ((Flags & 1) != 0)
  111. {
  112. Align = Context.Ns.Gpu.ReserveMemory(Align, (long)Pages * PageSize, 1);
  113. }
  114. else
  115. {
  116. Align = Context.Ns.Gpu.ReserveMemory((long)Pages * PageSize, Align);
  117. }
  118. Context.Memory.WriteInt64(Position + 0x10, Align);
  119. return 0;
  120. }
  121. private static long NvGpuAsIoctlMapBufferEx(ServiceCtx Context)
  122. {
  123. long Position = Context.Request.PtrBuff[0].Position;
  124. MemReader Reader = new MemReader(Context.Memory, Position);
  125. int Flags = Reader.ReadInt32();
  126. int Kind = Reader.ReadInt32();
  127. int Handle = Reader.ReadInt32();
  128. int PageSize = Reader.ReadInt32();
  129. long BuffAddr = Reader.ReadInt64();
  130. long MapSize = Reader.ReadInt64();
  131. long Offset = Reader.ReadInt64();
  132. HNvMap NvMap = Context.Ns.Os.Handles.GetData<HNvMap>(Handle);
  133. if (NvMap != null)
  134. {
  135. if ((Flags & 1) != 0)
  136. {
  137. Offset = Context.Ns.Gpu.MapMemory(NvMap.Address, Offset, NvMap.Size);
  138. }
  139. else
  140. {
  141. Offset = Context.Ns.Gpu.MapMemory(NvMap.Address, NvMap.Size);
  142. }
  143. }
  144. Context.Memory.WriteInt64(Position + 0x20, Offset);
  145. return 0;
  146. }
  147. private static long NvGpuAsIoctlGetVaRegions(ServiceCtx Context)
  148. {
  149. long Position = Context.Request.PtrBuff[0].Position;
  150. MemReader Reader = new MemReader(Context.Memory, Position);
  151. MemWriter Writer = new MemWriter(Context.Memory, Position);
  152. long Unused = Reader.ReadInt64();
  153. int BuffSize = Reader.ReadInt32();
  154. int Padding = Reader.ReadInt32();
  155. BuffSize = 0x30;
  156. Writer.WriteInt64(Unused);
  157. Writer.WriteInt32(BuffSize);
  158. Writer.WriteInt32(Padding);
  159. Writer.WriteInt64(0);
  160. Writer.WriteInt32(0);
  161. Writer.WriteInt32(0);
  162. Writer.WriteInt64(0);
  163. Writer.WriteInt64(0);
  164. Writer.WriteInt32(0);
  165. Writer.WriteInt32(0);
  166. Writer.WriteInt64(0);
  167. return 0;
  168. }
  169. private static long NvGpuAsIoctlInitializeEx(ServiceCtx Context)
  170. {
  171. long Position = Context.Request.PtrBuff[0].Position;
  172. MemReader Reader = new MemReader(Context.Memory, Position);
  173. int BigPageSize = Reader.ReadInt32();
  174. int AsFd = Reader.ReadInt32();
  175. int Flags = Reader.ReadInt32();
  176. int Reserved = Reader.ReadInt32();
  177. long Unknown10 = Reader.ReadInt64();
  178. long Unknown18 = Reader.ReadInt64();
  179. long Unknown20 = Reader.ReadInt64();
  180. return 0;
  181. }
  182. private static long NvHostIoctlCtrlGetConfig(ServiceCtx Context)
  183. {
  184. long Position = Context.Request.PtrBuff[0].Position;
  185. MemReader Reader = new MemReader(Context.Memory, Position);
  186. MemWriter Writer = new MemWriter(Context.Memory, Position + 0x82);
  187. for (int Index = 0; Index < 0x101; Index++)
  188. {
  189. Writer.WriteByte(0);
  190. }
  191. return 0;
  192. }
  193. private static long NvHostIoctlCtrlEventWait(ServiceCtx Context)
  194. {
  195. long Position = Context.Request.PtrBuff[0].Position;
  196. MemReader Reader = new MemReader(Context.Memory, Position);
  197. int SyncPtId = Reader.ReadInt32();
  198. int Threshold = Reader.ReadInt32();
  199. int Timeout = Reader.ReadInt32();
  200. int Value = Reader.ReadInt32();
  201. Context.Memory.WriteInt32(Position + 0xc, 0xcafe);
  202. return 0;
  203. }
  204. private static long NvGpuIoctlZcullGetCtxSize(ServiceCtx Context)
  205. {
  206. long Position = Context.Request.PtrBuff[0].Position;
  207. Context.Memory.WriteInt32(Position, 1);
  208. return 0;
  209. }
  210. private static long NvGpuIoctlZcullGetInfo(ServiceCtx Context)
  211. {
  212. long Position = Context.Request.PtrBuff[0].Position;
  213. MemWriter Writer = new MemWriter(Context.Memory, Position);
  214. Writer.WriteInt32(0);
  215. Writer.WriteInt32(0);
  216. Writer.WriteInt32(0);
  217. Writer.WriteInt32(0);
  218. Writer.WriteInt32(0);
  219. Writer.WriteInt32(0);
  220. Writer.WriteInt32(0);
  221. Writer.WriteInt32(0);
  222. Writer.WriteInt32(0);
  223. Writer.WriteInt32(0);
  224. return 0;
  225. }
  226. private static long NvGpuIoctlGetCharacteristics(ServiceCtx Context)
  227. {
  228. long Position = Context.Request.PtrBuff[0].Position;
  229. MemReader Reader = new MemReader(Context.Memory, Position);
  230. MemWriter Writer = new MemWriter(Context.Memory, Position);
  231. //Note: We should just ignore the BuffAddr, because official code
  232. //does __memcpy_device from Position + 0x10 to BuffAddr.
  233. long BuffSize = Reader.ReadInt64();
  234. long BuffAddr = Reader.ReadInt64();
  235. BuffSize = 0xa0;
  236. Writer.WriteInt64(BuffSize);
  237. Writer.WriteInt64(BuffAddr);
  238. Writer.WriteInt32(0x120); //NVGPU_GPU_ARCH_GM200
  239. Writer.WriteInt32(0xb); //NVGPU_GPU_IMPL_GM20B
  240. Writer.WriteInt32(0xa1);
  241. Writer.WriteInt32(1);
  242. Writer.WriteInt64(0x40000);
  243. Writer.WriteInt64(0);
  244. Writer.WriteInt32(2);
  245. Writer.WriteInt32(0x20); //NVGPU_GPU_BUS_TYPE_AXI
  246. Writer.WriteInt32(0x20000);
  247. Writer.WriteInt32(0x20000);
  248. Writer.WriteInt32(0x1b);
  249. Writer.WriteInt32(0x30000);
  250. Writer.WriteInt32(1);
  251. Writer.WriteInt32(0x503);
  252. Writer.WriteInt32(0x503);
  253. Writer.WriteInt32(0x80);
  254. Writer.WriteInt32(0x28);
  255. Writer.WriteInt32(0);
  256. Writer.WriteInt64(0x55);
  257. Writer.WriteInt32(0x902d); //FERMI_TWOD_A
  258. Writer.WriteInt32(0xb197); //MAXWELL_B
  259. Writer.WriteInt32(0xb1c0); //MAXWELL_COMPUTE_B
  260. Writer.WriteInt32(0xb06f); //MAXWELL_CHANNEL_GPFIFO_A
  261. Writer.WriteInt32(0xa140); //KEPLER_INLINE_TO_MEMORY_B
  262. Writer.WriteInt32(0xb0b5); //MAXWELL_DMA_COPY_A
  263. Writer.WriteInt32(1);
  264. Writer.WriteInt32(0);
  265. Writer.WriteInt32(2);
  266. Writer.WriteInt32(1);
  267. Writer.WriteInt32(0);
  268. Writer.WriteInt32(1);
  269. Writer.WriteInt32(0x21d70);
  270. Writer.WriteInt32(0);
  271. Writer.WriteByte((byte)'g');
  272. Writer.WriteByte((byte)'m');
  273. Writer.WriteByte((byte)'2');
  274. Writer.WriteByte((byte)'0');
  275. Writer.WriteByte((byte)'b');
  276. Writer.WriteByte((byte)'\0');
  277. Writer.WriteByte((byte)'\0');
  278. Writer.WriteByte((byte)'\0');
  279. Writer.WriteInt64(0);
  280. return 0;
  281. }
  282. private static long NvGpuIoctlGetTpcMasks(ServiceCtx Context)
  283. {
  284. long Position = Context.Request.PtrBuff[0].Position;
  285. MemReader Reader = new MemReader(Context.Memory, Position);
  286. int MaskBuffSize = Reader.ReadInt32();
  287. int Reserved = Reader.ReadInt32();
  288. long MaskBuffAddr = Reader.ReadInt64();
  289. long Unknown = Reader.ReadInt64();
  290. return 0;
  291. }
  292. private static long NvGpuIoctlZbcGetActiveSlotMask(ServiceCtx Context)
  293. {
  294. long Position = Context.Request.PtrBuff[0].Position;
  295. Context.Memory.WriteInt32(Position + 0, 7);
  296. Context.Memory.WriteInt32(Position + 4, 1);
  297. return 0;
  298. }
  299. private static long NvMapIoctlChannelSetUserData(ServiceCtx Context)
  300. {
  301. long Position = Context.Request.PtrBuff[0].Position;
  302. return 0;
  303. }
  304. private static long NvMapIoctlChannelSetNvMap(ServiceCtx Context)
  305. {
  306. long Position = Context.Request.PtrBuff[0].Position;
  307. int Fd = Context.Memory.ReadInt32(Position);
  308. return 0;
  309. }
  310. private static long NvMapIoctlChannelSubmitGpFifo(ServiceCtx Context)
  311. {
  312. long Position = Context.Request.PtrBuff[0].Position;
  313. MemReader Reader = new MemReader(Context.Memory, Position);
  314. MemWriter Writer = new MemWriter(Context.Memory, Position + 0x10);
  315. long GpFifo = Reader.ReadInt64();
  316. int Count = Reader.ReadInt32();
  317. int Flags = Reader.ReadInt32();
  318. int FenceId = Reader.ReadInt32();
  319. int FenceVal = Reader.ReadInt32();
  320. for (int Index = 0; Index < Count; Index++)
  321. {
  322. long GpFifoHdr = Reader.ReadInt64();
  323. long GpuAddr = GpFifoHdr & 0xffffffffff;
  324. int Size = (int)(GpFifoHdr >> 40) & 0x7ffffc;
  325. long CpuAddr = Context.Ns.Gpu.GetCpuAddr(GpuAddr);
  326. if (CpuAddr != -1)
  327. {
  328. byte[] Data = AMemoryHelper.ReadBytes(Context.Memory, CpuAddr, Size);
  329. NsGpuPBEntry[] PushBuffer = NsGpuPBEntry.DecodePushBuffer(Data);
  330. Context.Ns.Gpu.ProcessPushBuffer(PushBuffer, Context.Memory);
  331. }
  332. }
  333. Writer.WriteInt32(0);
  334. Writer.WriteInt32(0);
  335. return 0;
  336. }
  337. private static long NvMapIoctlChannelAllocObjCtx(ServiceCtx Context)
  338. {
  339. long Position = Context.Request.PtrBuff[0].Position;
  340. int ClassNum = Context.Memory.ReadInt32(Position + 0);
  341. int Flags = Context.Memory.ReadInt32(Position + 4);
  342. Context.Memory.WriteInt32(Position + 8, 0);
  343. return 0;
  344. }
  345. private static long NvMapIoctlChannelZcullBind(ServiceCtx Context)
  346. {
  347. long Position = Context.Request.PtrBuff[0].Position;
  348. MemReader Reader = new MemReader(Context.Memory, Position);
  349. long GpuVa = Reader.ReadInt64();
  350. int Mode = Reader.ReadInt32();
  351. int Padding = Reader.ReadInt32();
  352. return 0;
  353. }
  354. private static long NvMapIoctlChannelSetErrorNotifier(ServiceCtx Context)
  355. {
  356. long Position = Context.Request.PtrBuff[0].Position;
  357. MemReader Reader = new MemReader(Context.Memory, Position);
  358. long Offset = Reader.ReadInt64();
  359. long Size = Reader.ReadInt64();
  360. int Mem = Reader.ReadInt32();
  361. int Padding = Reader.ReadInt32();
  362. return 0;
  363. }
  364. private static long NvMapIoctlChannelSetPriority(ServiceCtx Context)
  365. {
  366. long Position = Context.Request.PtrBuff[0].Position;
  367. int Priority = Context.Memory.ReadInt32(Position);
  368. return 0;
  369. }
  370. private static long NvMapIoctlChannelAllocGpFifoEx2(ServiceCtx Context)
  371. {
  372. long Position = Context.Request.PtrBuff[0].Position;
  373. MemReader Reader = new MemReader(Context.Memory, Position);
  374. MemWriter Writer = new MemWriter(Context.Memory, Position + 0xc);
  375. int Count = Reader.ReadInt32();
  376. int Flags = Reader.ReadInt32();
  377. int Unknown8 = Reader.ReadInt32();
  378. long Fence = Reader.ReadInt64();
  379. int Unknown14 = Reader.ReadInt32();
  380. int Unknown18 = Reader.ReadInt32();
  381. Writer.WriteInt32(0);
  382. Writer.WriteInt32(0);
  383. return 0;
  384. }
  385. private static long NvMapIocCreate(ServiceCtx Context)
  386. {
  387. long Position = Context.Request.GetSendBuffPtr();
  388. int Size = Context.Memory.ReadInt32(Position);
  389. int Id = Context.Ns.Os.NvMapIds.GenerateId();
  390. int Handle = Context.Ns.Os.Handles.GenerateId(new HNvMap(Id, Size));
  391. Context.Memory.WriteInt32(Position + 4, Handle);
  392. return 0;
  393. }
  394. private static long NvMapIocFromId(ServiceCtx Context)
  395. {
  396. long Position = Context.Request.GetSendBuffPtr();
  397. int Id = Context.Memory.ReadInt32(Position);
  398. int Handle = -1;
  399. foreach (KeyValuePair<int, object> KV in Context.Ns.Os.Handles)
  400. {
  401. if (KV.Value is HNvMap NvMap && NvMap.Id == Id)
  402. {
  403. Handle = KV.Key;
  404. break;
  405. }
  406. }
  407. Context.Memory.WriteInt32(Position + 4, Handle);
  408. return 0;
  409. }
  410. private static long NvMapIocAlloc(ServiceCtx Context)
  411. {
  412. long Position = Context.Request.GetSendBuffPtr();
  413. MemReader Reader = new MemReader(Context.Memory, Position);
  414. int Handle = Reader.ReadInt32();
  415. int HeapMask = Reader.ReadInt32();
  416. int Flags = Reader.ReadInt32();
  417. int Align = Reader.ReadInt32();
  418. byte Kind = (byte)Reader.ReadInt64();
  419. long Addr = Reader.ReadInt64();
  420. HNvMap NvMap = Context.Ns.Os.Handles.GetData<HNvMap>(Handle);
  421. if (NvMap != null)
  422. {
  423. NvMap.Address = Addr;
  424. NvMap.Align = Align;
  425. NvMap.Kind = Kind;
  426. }
  427. Logging.Debug($"NvMapIocAlloc at {NvMap.Address:x16}");
  428. return 0;
  429. }
  430. private static long NvMapIocParam(ServiceCtx Context)
  431. {
  432. long Position = Context.Request.GetSendBuffPtr();
  433. MemReader Reader = new MemReader(Context.Memory, Position);
  434. int Handle = Reader.ReadInt32();
  435. int Param = Reader.ReadInt32();
  436. HNvMap NvMap = Context.Ns.Os.Handles.GetData<HNvMap>(Handle);
  437. int Response = 0;
  438. switch (Param)
  439. {
  440. case 1: Response = NvMap.Size; break;
  441. case 2: Response = NvMap.Align; break;
  442. case 4: Response = 0x40000000; break;
  443. case 5: Response = NvMap.Kind; break;
  444. }
  445. Context.Memory.WriteInt32(Position + 8, Response);
  446. return 0;
  447. }
  448. private static long NvMapIocGetId(ServiceCtx Context)
  449. {
  450. long Position = Context.Request.GetSendBuffPtr();
  451. int Handle = Context.Memory.ReadInt32(Position + 4);
  452. HNvMap NvMap = Context.Ns.Os.Handles.GetData<HNvMap>(Handle);
  453. Context.Memory.WriteInt32(Position, NvMap.Id);
  454. return 0;
  455. }
  456. }
  457. }