SDL_render_psp.c 44 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436
  1. /*
  2. Simple DirectMedia Layer
  3. Copyright (C) 1997-2025 Sam Lantinga <[email protected]>
  4. This software is provided 'as-is', without any express or implied
  5. warranty. In no event will the authors be held liable for any damages
  6. arising from the use of this software.
  7. Permission is granted to anyone to use this software for any purpose,
  8. including commercial applications, and to alter it and redistribute it
  9. freely, subject to the following restrictions:
  10. 1. The origin of this software must not be misrepresented; you must not
  11. claim that you wrote the original software. If you use this software
  12. in a product, an acknowledgment in the product documentation would be
  13. appreciated but is not required.
  14. 2. Altered source versions must be plainly marked as such, and must not be
  15. misrepresented as being the original software.
  16. 3. This notice may not be removed or altered from any source distribution.
  17. */
  18. #include "SDL_internal.h"
  19. #ifdef SDL_VIDEO_RENDER_PSP
  20. #include "../SDL_sysrender.h"
  21. #include "SDL_render_psp_c.h"
  22. #include <pspkernel.h>
  23. #include <pspdisplay.h>
  24. #include <pspgu.h>
  25. #include <pspgum.h>
  26. #include <stdio.h>
  27. #include <string.h>
  28. #include <math.h>
  29. #include <pspge.h>
  30. #include <stdarg.h>
  31. #include <stdlib.h>
  32. #include <vram.h>
  33. // PSP renderer implementation, based on the PGE
  34. static unsigned int __attribute__((aligned(16))) DisplayList[262144];
  35. #define COL5650(r, g, b, a) ((r >> 3) | ((g >> 2) << 5) | ((b >> 3) << 11))
  36. #define COL5551(r, g, b, a) ((r >> 3) | ((g >> 3) << 5) | ((b >> 3) << 10) | (a > 0 ? 0x7000 : 0))
  37. #define COL4444(r, g, b, a) ((r >> 4) | ((g >> 4) << 4) | ((b >> 4) << 8) | ((a >> 4) << 12))
  38. #define COL8888(r, g, b, a) ((r) | ((g) << 8) | ((b) << 16) | ((a) << 24))
  39. /**
  40. * Holds psp specific texture data
  41. *
  42. * Part of a hot-list of textures that are used as render targets
  43. * When short of vram we spill Least-Recently-Used render targets to system memory
  44. */
  45. typedef struct PSP_TextureData
  46. {
  47. void *data; /**< Image data. */
  48. unsigned int size; /**< Size of data in bytes. */
  49. unsigned int width; /**< Image width. */
  50. unsigned int height; /**< Image height. */
  51. unsigned int textureWidth; /**< Texture width (power of two). */
  52. unsigned int textureHeight; /**< Texture height (power of two). */
  53. unsigned int bits; /**< Image bits per pixel. */
  54. unsigned int format; /**< Image format - one of ::pgePixelFormat. */
  55. unsigned int pitch;
  56. bool swizzled; /**< Is image swizzled. */
  57. struct PSP_TextureData *prevhotw; /**< More recently used render target */
  58. struct PSP_TextureData *nexthotw; /**< Less recently used render target */
  59. } PSP_TextureData;
  60. typedef struct
  61. {
  62. SDL_BlendMode mode;
  63. unsigned int color;
  64. int shadeModel;
  65. SDL_Texture *texture;
  66. SDL_ScaleMode texture_scale_mode;
  67. SDL_TextureAddressMode texture_address_mode;
  68. } PSP_BlendState;
  69. typedef struct
  70. {
  71. unsigned int color;
  72. } PSP_DrawStateCache;
  73. typedef struct
  74. {
  75. void *frontbuffer; /**< main screen buffer */
  76. void *backbuffer; /**< buffer presented to display */
  77. SDL_Texture *boundTarget; /**< currently bound rendertarget */
  78. bool initialized; /**< is driver initialized */
  79. bool displayListAvail; /**< is the display list already initialized for this frame */
  80. unsigned int psm; /**< format of the display buffers */
  81. unsigned int bpp; /**< bits per pixel of the main display */
  82. bool vsync; /**< whether we do vsync */
  83. PSP_BlendState blendState; /**< current blend mode */
  84. PSP_TextureData *most_recent_target; /**< start of render target LRU double linked list */
  85. PSP_TextureData *least_recent_target; /**< end of the LRU list */
  86. bool vblank_not_reached; /**< whether vblank wasn't reached */
  87. } PSP_RenderData;
  88. typedef struct
  89. {
  90. float x, y, z;
  91. } VertV;
  92. typedef struct
  93. {
  94. float u, v;
  95. float x, y, z;
  96. } VertTV;
  97. typedef struct
  98. {
  99. SDL_Color col;
  100. float x, y, z;
  101. } VertCV;
  102. typedef struct
  103. {
  104. float u, v;
  105. SDL_Color col;
  106. float x, y, z;
  107. } VertTCV;
  108. #define radToDeg(x) ((x)*180.f / SDL_PI_F)
  109. #define degToRad(x) ((x)*SDL_PI_F / 180.f)
  110. static float MathAbs(float x)
  111. {
  112. float result;
  113. __asm__ volatile(
  114. "mtv %1, S000\n"
  115. "vabs.s S000, S000\n"
  116. "mfv %0, S000\n"
  117. : "=r"(result)
  118. : "r"(x));
  119. return result;
  120. }
  121. static void MathSincos(float r, float *s, float *c)
  122. {
  123. __asm__ volatile(
  124. "mtv %2, S002\n"
  125. "vcst.s S003, VFPU_2_PI\n"
  126. "vmul.s S002, S002, S003\n"
  127. "vrot.p C000, S002, [s, c]\n"
  128. "mfv %0, S000\n"
  129. "mfv %1, S001\n"
  130. : "=r"(*s), "=r"(*c)
  131. : "r"(r));
  132. }
  133. static void Swap(float *a, float *b)
  134. {
  135. float n = *a;
  136. *a = *b;
  137. *b = n;
  138. }
  139. static inline int InVram(void *data)
  140. {
  141. return data < (void *)0x04200000;
  142. }
  143. // Return next power of 2
  144. static int TextureNextPow2(unsigned int w)
  145. {
  146. unsigned int n = 2;
  147. if (w == 0) {
  148. return 0;
  149. }
  150. while (w > n) {
  151. n <<= 1;
  152. }
  153. return n;
  154. }
  155. static void psp_on_vblank(u32 sub, PSP_RenderData *data)
  156. {
  157. if (data) {
  158. data->vblank_not_reached = false;
  159. }
  160. }
  161. static int PixelFormatToPSPFMT(SDL_PixelFormat format)
  162. {
  163. switch (format) {
  164. case SDL_PIXELFORMAT_BGR565:
  165. return GU_PSM_5650;
  166. case SDL_PIXELFORMAT_ABGR1555:
  167. return GU_PSM_5551;
  168. case SDL_PIXELFORMAT_ABGR4444:
  169. return GU_PSM_4444;
  170. case SDL_PIXELFORMAT_ABGR8888:
  171. return GU_PSM_8888;
  172. default:
  173. return GU_PSM_8888;
  174. }
  175. }
  176. /// SECTION render target LRU management
  177. static void LRUTargetRelink(PSP_TextureData *psp_texture)
  178. {
  179. if (psp_texture->prevhotw) {
  180. psp_texture->prevhotw->nexthotw = psp_texture->nexthotw;
  181. }
  182. if (psp_texture->nexthotw) {
  183. psp_texture->nexthotw->prevhotw = psp_texture->prevhotw;
  184. }
  185. }
  186. static void LRUTargetPushFront(PSP_RenderData *data, PSP_TextureData *psp_texture)
  187. {
  188. psp_texture->nexthotw = data->most_recent_target;
  189. if (data->most_recent_target) {
  190. data->most_recent_target->prevhotw = psp_texture;
  191. }
  192. data->most_recent_target = psp_texture;
  193. if (!data->least_recent_target) {
  194. data->least_recent_target = psp_texture;
  195. }
  196. }
  197. static void LRUTargetRemove(PSP_RenderData *data, PSP_TextureData *psp_texture)
  198. {
  199. LRUTargetRelink(psp_texture);
  200. if (data->most_recent_target == psp_texture) {
  201. data->most_recent_target = psp_texture->nexthotw;
  202. }
  203. if (data->least_recent_target == psp_texture) {
  204. data->least_recent_target = psp_texture->prevhotw;
  205. }
  206. psp_texture->prevhotw = NULL;
  207. psp_texture->nexthotw = NULL;
  208. }
  209. static void LRUTargetBringFront(PSP_RenderData *data, PSP_TextureData *psp_texture)
  210. {
  211. if (data->most_recent_target == psp_texture) {
  212. return; // nothing to do
  213. }
  214. LRUTargetRemove(data, psp_texture);
  215. LRUTargetPushFront(data, psp_texture);
  216. }
  217. static void TextureStorageFree(void *storage)
  218. {
  219. if (InVram(storage)) {
  220. vfree(storage);
  221. } else {
  222. SDL_free(storage);
  223. }
  224. }
  225. static bool TextureSwizzle(PSP_TextureData *psp_texture, void *dst)
  226. {
  227. int bytewidth, height;
  228. int rowblocks, rowblocksadd;
  229. int i, j;
  230. unsigned int blockaddress = 0;
  231. unsigned int *src = NULL;
  232. unsigned char *data = NULL;
  233. if (psp_texture->swizzled) {
  234. return true;
  235. }
  236. bytewidth = psp_texture->textureWidth * (psp_texture->bits >> 3);
  237. height = psp_texture->size / bytewidth;
  238. rowblocks = (bytewidth >> 4);
  239. rowblocksadd = (rowblocks - 1) << 7;
  240. src = (unsigned int *)psp_texture->data;
  241. data = dst;
  242. if (!data) {
  243. data = SDL_malloc(psp_texture->size);
  244. }
  245. if (!data) {
  246. return false;
  247. }
  248. for (j = 0; j < height; j++, blockaddress += 16) {
  249. unsigned int *block;
  250. block = (unsigned int *)&data[blockaddress];
  251. for (i = 0; i < rowblocks; i++) {
  252. *block++ = *src++;
  253. *block++ = *src++;
  254. *block++ = *src++;
  255. *block++ = *src++;
  256. block += 28;
  257. }
  258. if ((j & 0x7) == 0x7) {
  259. blockaddress += rowblocksadd;
  260. }
  261. }
  262. TextureStorageFree(psp_texture->data);
  263. psp_texture->data = data;
  264. psp_texture->swizzled = true;
  265. sceKernelDcacheWritebackRange(psp_texture->data, psp_texture->size);
  266. return true;
  267. }
  268. static bool TextureUnswizzle(PSP_TextureData *psp_texture, void *dst)
  269. {
  270. int bytewidth, height;
  271. int widthblocks, heightblocks;
  272. int dstpitch, dstrow;
  273. int blockx, blocky;
  274. int j;
  275. unsigned int *src = NULL;
  276. unsigned char *data = NULL;
  277. unsigned char *ydst = NULL;
  278. if (!psp_texture->swizzled) {
  279. return true;
  280. }
  281. bytewidth = psp_texture->textureWidth * (psp_texture->bits >> 3);
  282. height = psp_texture->size / bytewidth;
  283. widthblocks = bytewidth / 16;
  284. heightblocks = height / 8;
  285. dstpitch = (bytewidth - 16) / 4;
  286. dstrow = bytewidth * 8;
  287. src = (unsigned int *)psp_texture->data;
  288. data = dst;
  289. if (!data) {
  290. data = SDL_malloc(psp_texture->size);
  291. }
  292. if (!data) {
  293. return false;
  294. }
  295. ydst = (unsigned char *)data;
  296. for (blocky = 0; blocky < heightblocks; ++blocky) {
  297. unsigned char *xdst = ydst;
  298. for (blockx = 0; blockx < widthblocks; ++blockx) {
  299. unsigned int *block;
  300. block = (unsigned int *)xdst;
  301. for (j = 0; j < 8; ++j) {
  302. *(block++) = *(src++);
  303. *(block++) = *(src++);
  304. *(block++) = *(src++);
  305. *(block++) = *(src++);
  306. block += dstpitch;
  307. }
  308. xdst += 16;
  309. }
  310. ydst += dstrow;
  311. }
  312. TextureStorageFree(psp_texture->data);
  313. psp_texture->data = data;
  314. psp_texture->swizzled = false;
  315. sceKernelDcacheWritebackRange(psp_texture->data, psp_texture->size);
  316. return true;
  317. }
  318. static bool TextureSpillToSram(PSP_RenderData *data, PSP_TextureData *psp_texture)
  319. {
  320. // Assumes the texture is in VRAM
  321. if (psp_texture->swizzled) {
  322. // Texture was swizzled in vram, just copy to system memory
  323. void *sdata = SDL_malloc(psp_texture->size);
  324. if (!sdata) {
  325. return false;
  326. }
  327. SDL_memcpy(sdata, psp_texture->data, psp_texture->size);
  328. vfree(psp_texture->data);
  329. psp_texture->data = sdata;
  330. return true;
  331. } else {
  332. return TextureSwizzle(psp_texture, NULL); // Will realloc in sysram
  333. }
  334. }
  335. static bool TexturePromoteToVram(PSP_RenderData *data, PSP_TextureData *psp_texture, bool target)
  336. {
  337. // Assumes texture in sram and a large enough continuous block in vram
  338. void *tdata = vramalloc(psp_texture->size);
  339. if (psp_texture->swizzled && target) {
  340. return TextureUnswizzle(psp_texture, tdata);
  341. } else {
  342. SDL_memcpy(tdata, psp_texture->data, psp_texture->size);
  343. SDL_free(psp_texture->data);
  344. psp_texture->data = tdata;
  345. return true;
  346. }
  347. }
  348. static bool TextureSpillLRU(PSP_RenderData *data, size_t wanted)
  349. {
  350. PSP_TextureData *lru = data->least_recent_target;
  351. if (lru) {
  352. if (!TextureSpillToSram(data, lru)) {
  353. return false;
  354. }
  355. LRUTargetRemove(data, lru);
  356. } else {
  357. // Asked to spill but there nothing to spill
  358. return SDL_SetError("Could not spill more VRAM to system memory. VRAM : %dKB,(%dKB), wanted %dKB", vmemavail() / 1024, vlargestblock() / 1024, wanted / 1024);
  359. }
  360. return true;
  361. }
  362. static bool TextureSpillTargetsForSpace(PSP_RenderData *data, size_t size)
  363. {
  364. while (vlargestblock() < size) {
  365. if (!TextureSpillLRU(data, size)) {
  366. return false;
  367. }
  368. }
  369. return true;
  370. }
  371. static bool TextureBindAsTarget(PSP_RenderData *data, PSP_TextureData *psp_texture)
  372. {
  373. unsigned int dstFormat;
  374. if (!InVram(psp_texture->data)) {
  375. // Bring back the texture in vram
  376. if (!TextureSpillTargetsForSpace(data, psp_texture->size)) {
  377. return false;
  378. }
  379. if (!TexturePromoteToVram(data, psp_texture, true)) {
  380. return false;
  381. }
  382. }
  383. LRUTargetBringFront(data, psp_texture);
  384. sceGuDrawBufferList(psp_texture->format, vrelptr(psp_texture->data), psp_texture->textureWidth);
  385. // Stencil alpha dst hack
  386. dstFormat = psp_texture->format;
  387. if (dstFormat == GU_PSM_5551) {
  388. sceGuEnable(GU_STENCIL_TEST);
  389. sceGuStencilOp(GU_REPLACE, GU_REPLACE, GU_REPLACE);
  390. sceGuStencilFunc(GU_GEQUAL, 0xff, 0xff);
  391. sceGuEnable(GU_ALPHA_TEST);
  392. sceGuAlphaFunc(GU_GREATER, 0x00, 0xff);
  393. } else {
  394. sceGuDisable(GU_STENCIL_TEST);
  395. sceGuDisable(GU_ALPHA_TEST);
  396. }
  397. return true;
  398. }
  399. static void PSP_WindowEvent(SDL_Renderer *renderer, const SDL_WindowEvent *event)
  400. {
  401. }
  402. static bool PSP_CreateTexture(SDL_Renderer *renderer, SDL_Texture *texture, SDL_PropertiesID create_props)
  403. {
  404. PSP_RenderData *data = renderer->internal;
  405. PSP_TextureData *psp_texture = (PSP_TextureData *)SDL_calloc(1, sizeof(*psp_texture));
  406. if (!psp_texture) {
  407. return false;
  408. }
  409. psp_texture->swizzled = false;
  410. psp_texture->width = texture->w;
  411. psp_texture->height = texture->h;
  412. psp_texture->textureHeight = TextureNextPow2(texture->h);
  413. psp_texture->textureWidth = TextureNextPow2(texture->w);
  414. psp_texture->format = PixelFormatToPSPFMT(texture->format);
  415. switch (psp_texture->format) {
  416. case GU_PSM_5650:
  417. case GU_PSM_5551:
  418. case GU_PSM_4444:
  419. psp_texture->bits = 16;
  420. break;
  421. case GU_PSM_8888:
  422. psp_texture->bits = 32;
  423. break;
  424. default:
  425. SDL_free(psp_texture);
  426. return false;
  427. }
  428. psp_texture->pitch = psp_texture->textureWidth * SDL_BYTESPERPIXEL(texture->format);
  429. psp_texture->size = psp_texture->textureHeight * psp_texture->pitch;
  430. if (texture->access == SDL_TEXTUREACCESS_TARGET) {
  431. if (!TextureSpillTargetsForSpace(renderer->internal, psp_texture->size)) {
  432. SDL_free(psp_texture);
  433. return false;
  434. }
  435. psp_texture->data = vramalloc(psp_texture->size);
  436. if (psp_texture->data) {
  437. LRUTargetPushFront(data, psp_texture);
  438. }
  439. } else {
  440. psp_texture->data = SDL_calloc(1, psp_texture->size);
  441. }
  442. if (!psp_texture->data) {
  443. SDL_free(psp_texture);
  444. return false;
  445. }
  446. texture->internal = psp_texture;
  447. return true;
  448. }
  449. static bool TextureShouldSwizzle(PSP_TextureData *psp_texture, SDL_Texture *texture)
  450. {
  451. return !((texture->access == SDL_TEXTUREACCESS_TARGET) && InVram(psp_texture->data)) && texture->access != SDL_TEXTUREACCESS_STREAMING && (texture->w >= 16 || texture->h >= 16);
  452. }
  453. static void SetTextureAddressMode(SDL_TextureAddressMode addressMode)
  454. {
  455. switch (addressMode) {
  456. case SDL_TEXTURE_ADDRESS_CLAMP:
  457. sceGuTexWrap(GU_CLAMP, GU_CLAMP);
  458. break;
  459. case SDL_TEXTURE_ADDRESS_WRAP:
  460. sceGuTexWrap(GU_REPEAT, GU_REPEAT);
  461. break;
  462. default:
  463. break;
  464. }
  465. }
  466. static void SetTextureScaleMode(SDL_ScaleMode scaleMode)
  467. {
  468. switch (scaleMode) {
  469. case SDL_SCALEMODE_NEAREST:
  470. sceGuTexFilter(GU_NEAREST, GU_NEAREST);
  471. break;
  472. case SDL_SCALEMODE_LINEAR:
  473. sceGuTexFilter(GU_LINEAR, GU_LINEAR);
  474. break;
  475. default:
  476. break;
  477. }
  478. }
  479. static void TextureActivate(SDL_Texture *texture)
  480. {
  481. PSP_TextureData *psp_texture = (PSP_TextureData *)texture->internal;
  482. // Swizzling is useless with small textures.
  483. if (TextureShouldSwizzle(psp_texture, texture)) {
  484. TextureSwizzle(psp_texture, NULL);
  485. }
  486. sceGuTexMode(psp_texture->format, 0, 0, psp_texture->swizzled);
  487. sceGuTexImage(0, psp_texture->textureWidth, psp_texture->textureHeight, psp_texture->textureWidth, psp_texture->data);
  488. }
  489. static bool PSP_LockTexture(SDL_Renderer *renderer, SDL_Texture *texture,
  490. const SDL_Rect *rect, void **pixels, int *pitch);
  491. static bool PSP_UpdateTexture(SDL_Renderer *renderer, SDL_Texture *texture,
  492. const SDL_Rect *rect, const void *pixels, int pitch)
  493. {
  494. /* PSP_TextureData *psp_texture = (PSP_TextureData *) texture->internal; */
  495. const Uint8 *src;
  496. Uint8 *dst;
  497. int row, length, dpitch;
  498. src = pixels;
  499. PSP_LockTexture(renderer, texture, rect, (void **)&dst, &dpitch);
  500. length = rect->w * SDL_BYTESPERPIXEL(texture->format);
  501. if (length == pitch && length == dpitch) {
  502. SDL_memcpy(dst, src, length * rect->h);
  503. } else {
  504. for (row = 0; row < rect->h; ++row) {
  505. SDL_memcpy(dst, src, length);
  506. src += pitch;
  507. dst += dpitch;
  508. }
  509. }
  510. sceKernelDcacheWritebackAll();
  511. return true;
  512. }
  513. static bool PSP_LockTexture(SDL_Renderer *renderer, SDL_Texture *texture,
  514. const SDL_Rect *rect, void **pixels, int *pitch)
  515. {
  516. PSP_TextureData *psp_texture = (PSP_TextureData *)texture->internal;
  517. *pixels =
  518. (void *)((Uint8 *)psp_texture->data + rect->y * psp_texture->pitch +
  519. rect->x * SDL_BYTESPERPIXEL(texture->format));
  520. *pitch = psp_texture->pitch;
  521. return true;
  522. }
  523. static void PSP_UnlockTexture(SDL_Renderer *renderer, SDL_Texture *texture)
  524. {
  525. PSP_TextureData *psp_texture = (PSP_TextureData *)texture->internal;
  526. SDL_Rect rect;
  527. // We do whole texture updates, at least for now
  528. rect.x = 0;
  529. rect.y = 0;
  530. rect.w = texture->w;
  531. rect.h = texture->h;
  532. PSP_UpdateTexture(renderer, texture, &rect, psp_texture->data, psp_texture->pitch);
  533. }
  534. static bool PSP_SetRenderTarget(SDL_Renderer *renderer, SDL_Texture *texture)
  535. {
  536. return true;
  537. }
  538. static bool PSP_QueueNoOp(SDL_Renderer *renderer, SDL_RenderCommand *cmd)
  539. {
  540. return true; // nothing to do in this backend.
  541. }
  542. static bool PSP_QueueDrawPoints(SDL_Renderer *renderer, SDL_RenderCommand *cmd, const SDL_FPoint *points, int count)
  543. {
  544. VertV *verts = (VertV *)SDL_AllocateRenderVertices(renderer, count * sizeof(VertV), 4, &cmd->data.draw.first);
  545. int i;
  546. if (!verts) {
  547. return false;
  548. }
  549. cmd->data.draw.count = count;
  550. for (i = 0; i < count; i++, verts++, points++) {
  551. verts->x = points->x;
  552. verts->y = points->y;
  553. verts->z = 0.0f;
  554. }
  555. return true;
  556. }
  557. static bool PSP_QueueGeometry(SDL_Renderer *renderer, SDL_RenderCommand *cmd, SDL_Texture *texture,
  558. const float *xy, int xy_stride, const SDL_FColor *color, int color_stride, const float *uv, int uv_stride,
  559. int num_vertices, const void *indices, int num_indices, int size_indices,
  560. float scale_x, float scale_y)
  561. {
  562. int i;
  563. int count = indices ? num_indices : num_vertices;
  564. const float color_scale = cmd->data.draw.color_scale;
  565. cmd->data.draw.count = count;
  566. size_indices = indices ? size_indices : 0;
  567. if (!texture) {
  568. VertCV *verts;
  569. verts = (VertCV *)SDL_AllocateRenderVertices(renderer, count * sizeof(VertCV), 4, &cmd->data.draw.first);
  570. if (!verts) {
  571. return false;
  572. }
  573. for (i = 0; i < count; i++) {
  574. int j;
  575. float *xy_;
  576. SDL_FColor *col_;
  577. if (size_indices == 4) {
  578. j = ((const Uint32 *)indices)[i];
  579. } else if (size_indices == 2) {
  580. j = ((const Uint16 *)indices)[i];
  581. } else if (size_indices == 1) {
  582. j = ((const Uint8 *)indices)[i];
  583. } else {
  584. j = i;
  585. }
  586. xy_ = (float *)((char *)xy + j * xy_stride);
  587. col_ = (SDL_FColor *)((char *)color + j * color_stride);
  588. verts->x = xy_[0] * scale_x;
  589. verts->y = xy_[1] * scale_y;
  590. verts->z = 0;
  591. verts->col.r = (Uint8)SDL_roundf(SDL_clamp(col_->r * color_scale, 0.0f, 1.0f) * 255.0f);
  592. verts->col.g = (Uint8)SDL_roundf(SDL_clamp(col_->g * color_scale, 0.0f, 1.0f) * 255.0f);
  593. verts->col.b = (Uint8)SDL_roundf(SDL_clamp(col_->b * color_scale, 0.0f, 1.0f) * 255.0f);
  594. verts->col.a = (Uint8)SDL_roundf(SDL_clamp(col_->a, 0.0f, 1.0f) * 255.0f);
  595. verts++;
  596. }
  597. } else {
  598. PSP_TextureData *psp_texture = (PSP_TextureData *)texture->internal;
  599. VertTCV *verts;
  600. verts = (VertTCV *)SDL_AllocateRenderVertices(renderer, count * sizeof(VertTCV), 4, &cmd->data.draw.first);
  601. if (!verts) {
  602. return false;
  603. }
  604. for (i = 0; i < count; i++) {
  605. int j;
  606. float *xy_;
  607. SDL_FColor *col_;
  608. float *uv_;
  609. if (size_indices == 4) {
  610. j = ((const Uint32 *)indices)[i];
  611. } else if (size_indices == 2) {
  612. j = ((const Uint16 *)indices)[i];
  613. } else if (size_indices == 1) {
  614. j = ((const Uint8 *)indices)[i];
  615. } else {
  616. j = i;
  617. }
  618. xy_ = (float *)((char *)xy + j * xy_stride);
  619. col_ = (SDL_FColor *)((char *)color + j * color_stride);
  620. uv_ = (float *)((char *)uv + j * uv_stride);
  621. verts->x = xy_[0] * scale_x;
  622. verts->y = xy_[1] * scale_y;
  623. verts->z = 0;
  624. verts->col.r = (Uint8)SDL_roundf(SDL_clamp(col_->r * color_scale, 0.0f, 1.0f) * 255.0f);
  625. verts->col.g = (Uint8)SDL_roundf(SDL_clamp(col_->g * color_scale, 0.0f, 1.0f) * 255.0f);
  626. verts->col.b = (Uint8)SDL_roundf(SDL_clamp(col_->b * color_scale, 0.0f, 1.0f) * 255.0f);
  627. verts->col.a = (Uint8)SDL_roundf(SDL_clamp(col_->a, 0.0f, 1.0f) * 255.0f);
  628. verts->u = uv_[0] * psp_texture->textureWidth;
  629. verts->v = uv_[1] * psp_texture->textureHeight;
  630. verts++;
  631. }
  632. }
  633. return true;
  634. }
  635. static bool PSP_QueueFillRects(SDL_Renderer *renderer, SDL_RenderCommand *cmd, const SDL_FRect *rects, int count)
  636. {
  637. VertV *verts = (VertV *)SDL_AllocateRenderVertices(renderer, count * 2 * sizeof(VertV), 4, &cmd->data.draw.first);
  638. int i;
  639. if (!verts) {
  640. return false;
  641. }
  642. cmd->data.draw.count = count;
  643. for (i = 0; i < count; i++, rects++) {
  644. verts->x = rects->x;
  645. verts->y = rects->y;
  646. verts->z = 0.0f;
  647. verts++;
  648. verts->x = rects->x + rects->w + 0.5f;
  649. verts->y = rects->y + rects->h + 0.5f;
  650. verts->z = 0.0f;
  651. verts++;
  652. }
  653. return true;
  654. }
  655. static bool PSP_QueueCopy(SDL_Renderer *renderer, SDL_RenderCommand *cmd, SDL_Texture *texture,
  656. const SDL_FRect *srcrect, const SDL_FRect *dstrect)
  657. {
  658. VertTV *verts;
  659. const float x = dstrect->x;
  660. const float y = dstrect->y;
  661. const float width = dstrect->w;
  662. const float height = dstrect->h;
  663. const float u0 = srcrect->x;
  664. const float v0 = srcrect->y;
  665. const float u1 = srcrect->x + srcrect->w;
  666. const float v1 = srcrect->y + srcrect->h;
  667. if ((MathAbs(u1) - MathAbs(u0)) < 64.0f) {
  668. verts = (VertTV *)SDL_AllocateRenderVertices(renderer, 2 * sizeof(VertTV), 4, &cmd->data.draw.first);
  669. if (!verts) {
  670. return false;
  671. }
  672. cmd->data.draw.count = 1;
  673. verts->u = u0;
  674. verts->v = v0;
  675. verts->x = x;
  676. verts->y = y;
  677. verts->z = 0;
  678. verts++;
  679. verts->u = u1;
  680. verts->v = v1;
  681. verts->x = x + width;
  682. verts->y = y + height;
  683. verts->z = 0;
  684. verts++;
  685. } else {
  686. float start, end;
  687. float curU = u0;
  688. float curX = x;
  689. const float endX = x + width;
  690. const float slice = 64.0f;
  691. const size_t count = (size_t)SDL_ceilf(width / slice);
  692. size_t i;
  693. float ustep = (u1 - u0) / width * slice;
  694. if (ustep < 0.0f) {
  695. ustep = -ustep;
  696. }
  697. cmd->data.draw.count = count;
  698. verts = (VertTV *)SDL_AllocateRenderVertices(renderer, count * 2 * sizeof(VertTV), 4, &cmd->data.draw.first);
  699. if (!verts) {
  700. return false;
  701. }
  702. for (i = 0, start = 0, end = width; i < count; i++, start += slice) {
  703. const float polyWidth = ((curX + slice) > endX) ? (endX - curX) : slice;
  704. const float sourceWidth = ((curU + ustep) > u1) ? (u1 - curU) : ustep;
  705. SDL_assert(start < end);
  706. verts->u = curU;
  707. verts->v = v0;
  708. verts->x = curX;
  709. verts->y = y;
  710. verts->z = 0;
  711. verts++;
  712. curU += sourceWidth;
  713. curX += polyWidth;
  714. verts->u = curU;
  715. verts->v = v1;
  716. verts->x = curX;
  717. verts->y = (y + height);
  718. verts->z = 0;
  719. verts++;
  720. }
  721. }
  722. return true;
  723. }
  724. static bool PSP_QueueCopyEx(SDL_Renderer *renderer, SDL_RenderCommand *cmd, SDL_Texture *texture,
  725. const SDL_FRect *srcrect, const SDL_FRect *dstrect,
  726. const double angle, const SDL_FPoint *center, const SDL_FlipMode flip, float scale_x, float scale_y)
  727. {
  728. VertTV *verts = (VertTV *)SDL_AllocateRenderVertices(renderer, 4 * sizeof(VertTV), 4, &cmd->data.draw.first);
  729. const float centerx = center->x;
  730. const float centery = center->y;
  731. const float x = dstrect->x + centerx;
  732. const float y = dstrect->y + centery;
  733. const float width = dstrect->w - centerx;
  734. const float height = dstrect->h - centery;
  735. float s, c;
  736. float cw1, sw1, ch1, sh1, cw2, sw2, ch2, sh2;
  737. float u0 = srcrect->x;
  738. float v0 = srcrect->y;
  739. float u1 = srcrect->x + srcrect->w;
  740. float v1 = srcrect->y + srcrect->h;
  741. if (!verts) {
  742. return false;
  743. }
  744. cmd->data.draw.count = 1;
  745. MathSincos(degToRad((float)(360 - angle)), &s, &c);
  746. cw1 = c * -centerx;
  747. sw1 = s * -centerx;
  748. ch1 = c * -centery;
  749. sh1 = s * -centery;
  750. cw2 = c * width;
  751. sw2 = s * width;
  752. ch2 = c * height;
  753. sh2 = s * height;
  754. if (flip & SDL_FLIP_VERTICAL) {
  755. Swap(&v0, &v1);
  756. }
  757. if (flip & SDL_FLIP_HORIZONTAL) {
  758. Swap(&u0, &u1);
  759. }
  760. verts->u = u0;
  761. verts->v = v0;
  762. verts->x = x + cw1 + sh1;
  763. verts->y = y - sw1 + ch1;
  764. verts->z = 0;
  765. verts++;
  766. verts->u = u0;
  767. verts->v = v1;
  768. verts->x = x + cw1 + sh2;
  769. verts->y = y - sw1 + ch2;
  770. verts->z = 0;
  771. verts++;
  772. verts->u = u1;
  773. verts->v = v1;
  774. verts->x = x + cw2 + sh2;
  775. verts->y = y - sw2 + ch2;
  776. verts->z = 0;
  777. verts++;
  778. verts->u = u1;
  779. verts->v = v0;
  780. verts->x = x + cw2 + sh1;
  781. verts->y = y - sw2 + ch1;
  782. verts->z = 0;
  783. if (scale_x != 1.0f || scale_y != 1.0f) {
  784. verts->x *= scale_x;
  785. verts->y *= scale_y;
  786. verts--;
  787. verts->x *= scale_x;
  788. verts->y *= scale_y;
  789. verts--;
  790. verts->x *= scale_x;
  791. verts->y *= scale_y;
  792. verts--;
  793. verts->x *= scale_x;
  794. verts->y *= scale_y;
  795. }
  796. return true;
  797. }
  798. static void ResetBlendState(PSP_BlendState *state)
  799. {
  800. sceGuColor(0xffffffff);
  801. state->color = 0xffffffff;
  802. state->mode = SDL_BLENDMODE_INVALID;
  803. state->texture = NULL;
  804. sceGuDisable(GU_TEXTURE_2D);
  805. sceGuShadeModel(GU_SMOOTH);
  806. state->shadeModel = GU_SMOOTH;
  807. }
  808. static void StartDrawing(SDL_Renderer *renderer)
  809. {
  810. PSP_RenderData *data = (PSP_RenderData *)renderer->internal;
  811. // Check if we need to start GU displaylist
  812. if (!data->displayListAvail) {
  813. sceGuStart(GU_DIRECT, DisplayList);
  814. data->displayListAvail = true;
  815. // ResetBlendState(&data->blendState);
  816. }
  817. // Check if we need a draw buffer change
  818. if (renderer->target != data->boundTarget) {
  819. SDL_Texture *texture = renderer->target;
  820. if (texture) {
  821. PSP_TextureData *psp_texture = (PSP_TextureData *)texture->internal;
  822. // Set target, registering LRU
  823. TextureBindAsTarget(data, psp_texture);
  824. } else {
  825. // Set target back to screen
  826. sceGuDrawBufferList(data->psm, vrelptr(data->frontbuffer), PSP_FRAME_BUFFER_WIDTH);
  827. }
  828. data->boundTarget = texture;
  829. }
  830. }
  831. static void PSP_SetBlendState(PSP_RenderData *data, PSP_BlendState *state)
  832. {
  833. PSP_BlendState *current = &data->blendState;
  834. if (state->mode != current->mode) {
  835. switch (state->mode) {
  836. case SDL_BLENDMODE_NONE:
  837. sceGuTexFunc(GU_TFX_REPLACE, GU_TCC_RGBA);
  838. sceGuDisable(GU_BLEND);
  839. break;
  840. case SDL_BLENDMODE_BLEND:
  841. sceGuTexFunc(GU_TFX_MODULATE, GU_TCC_RGBA);
  842. sceGuBlendFunc(GU_ADD, GU_SRC_ALPHA, GU_ONE_MINUS_SRC_ALPHA, 0, 0);
  843. sceGuEnable(GU_BLEND);
  844. break;
  845. case SDL_BLENDMODE_BLEND_PREMULTIPLIED:
  846. sceGuTexFunc(GU_TFX_MODULATE , GU_TCC_RGBA);
  847. sceGuBlendFunc(GU_ADD, GU_FIX, GU_ONE_MINUS_SRC_ALPHA, 0x00FFFFFF, 0 );
  848. sceGuEnable(GU_BLEND);
  849. break;
  850. case SDL_BLENDMODE_ADD:
  851. sceGuTexFunc(GU_TFX_MODULATE, GU_TCC_RGBA);
  852. sceGuBlendFunc(GU_ADD, GU_SRC_ALPHA, GU_FIX, 0, 0x00FFFFFF);
  853. sceGuEnable(GU_BLEND);
  854. break;
  855. case SDL_BLENDMODE_ADD_PREMULTIPLIED:
  856. sceGuTexFunc(GU_TFX_MODULATE, GU_TCC_RGBA);
  857. sceGuBlendFunc(GU_ADD, GU_FIX, GU_FIX, 0, 0x00FFFFFF);
  858. sceGuEnable(GU_BLEND);
  859. break;
  860. case SDL_BLENDMODE_MOD:
  861. sceGuTexFunc(GU_TFX_MODULATE, GU_TCC_RGBA);
  862. sceGuBlendFunc(GU_ADD, GU_FIX, GU_SRC_COLOR, 0, 0);
  863. sceGuEnable(GU_BLEND);
  864. break;
  865. case SDL_BLENDMODE_MUL:
  866. sceGuTexFunc(GU_TFX_MODULATE, GU_TCC_RGBA);
  867. // FIXME SDL_BLENDMODE_MUL is simplified, and dstA is in fact un-changed.
  868. sceGuBlendFunc(GU_ADD, GU_DST_COLOR, GU_ONE_MINUS_SRC_ALPHA, 0, 0);
  869. sceGuEnable(GU_BLEND);
  870. break;
  871. case SDL_BLENDMODE_INVALID:
  872. break;
  873. }
  874. }
  875. if (state->color != current->color) {
  876. sceGuColor(state->color);
  877. }
  878. if (state->shadeModel != current->shadeModel) {
  879. sceGuShadeModel(state->shadeModel);
  880. }
  881. if (state->texture != current->texture) {
  882. if (state->texture) {
  883. TextureActivate(state->texture);
  884. sceGuEnable(GU_TEXTURE_2D);
  885. } else {
  886. sceGuDisable(GU_TEXTURE_2D);
  887. }
  888. }
  889. if (state->texture) {
  890. SetTextureScaleMode(state->texture_scale_mode);
  891. SetTextureAddressMode(state->texture_address_mode);
  892. }
  893. *current = *state;
  894. }
  895. static void PSP_InvalidateCachedState(SDL_Renderer *renderer)
  896. {
  897. // currently this doesn't do anything. If this needs to do something (and someone is mixing their own rendering calls in!), update this.
  898. }
  899. static bool PSP_RunCommandQueue(SDL_Renderer *renderer, SDL_RenderCommand *cmd, void *vertices, size_t vertsize)
  900. {
  901. PSP_RenderData *data = (PSP_RenderData *)renderer->internal;
  902. Uint8 *gpumem = NULL;
  903. PSP_DrawStateCache drawstate;
  904. drawstate.color = 0;
  905. StartDrawing(renderer);
  906. /* note that before the renderer interface change, this would do extremely small
  907. batches with sceGuGetMemory()--a few vertices at a time--and it's not clear that
  908. this won't fail if you try to push 100,000 draw calls in a single batch.
  909. I don't know what the limits on PSP hardware are. It might be useful to have
  910. rendering backends report a reasonable maximum, so the higher level can flush
  911. if we appear to be exceeding that. */
  912. gpumem = (Uint8 *)sceGuGetMemory(vertsize);
  913. if (!gpumem) {
  914. return SDL_SetError("Couldn't obtain a %d-byte vertex buffer!", (int)vertsize);
  915. }
  916. SDL_memcpy(gpumem, vertices, vertsize);
  917. while (cmd) {
  918. switch (cmd->command) {
  919. case SDL_RENDERCMD_SETDRAWCOLOR:
  920. {
  921. const Uint8 r = (Uint8)SDL_roundf(SDL_clamp(cmd->data.color.color.r * cmd->data.color.color_scale, 0.0f, 1.0f) * 255.0f);
  922. const Uint8 g = (Uint8)SDL_roundf(SDL_clamp(cmd->data.color.color.g * cmd->data.color.color_scale, 0.0f, 1.0f) * 255.0f);
  923. const Uint8 b = (Uint8)SDL_roundf(SDL_clamp(cmd->data.color.color.b * cmd->data.color.color_scale, 0.0f, 1.0f) * 255.0f);
  924. const Uint8 a = (Uint8)SDL_roundf(SDL_clamp(cmd->data.color.color.a, 0.0f, 1.0f) * 255.0f);
  925. drawstate.color = GU_RGBA(r, g, b, a);
  926. break;
  927. }
  928. case SDL_RENDERCMD_SETVIEWPORT:
  929. {
  930. SDL_Rect *viewport = &cmd->data.viewport.rect;
  931. sceGuOffset(2048 - (viewport->w >> 1), 2048 - (viewport->h >> 1));
  932. sceGuViewport(2048, 2048, viewport->w, viewport->h);
  933. sceGuScissor(viewport->x, viewport->y, viewport->w, viewport->h);
  934. // FIXME: We need to update the clip rect too, see https://github.com/libsdl-org/SDL/issues/9094
  935. break;
  936. }
  937. case SDL_RENDERCMD_SETCLIPRECT:
  938. {
  939. const SDL_Rect *rect = &cmd->data.cliprect.rect;
  940. if (cmd->data.cliprect.enabled) {
  941. sceGuEnable(GU_SCISSOR_TEST);
  942. sceGuScissor(rect->x, rect->y, rect->w, rect->h);
  943. } else {
  944. sceGuDisable(GU_SCISSOR_TEST);
  945. }
  946. break;
  947. }
  948. case SDL_RENDERCMD_CLEAR:
  949. {
  950. const Uint8 r = (Uint8)SDL_roundf(SDL_clamp(cmd->data.color.color.r * cmd->data.color.color_scale, 0.0f, 1.0f) * 255.0f);
  951. const Uint8 g = (Uint8)SDL_roundf(SDL_clamp(cmd->data.color.color.g * cmd->data.color.color_scale, 0.0f, 1.0f) * 255.0f);
  952. const Uint8 b = (Uint8)SDL_roundf(SDL_clamp(cmd->data.color.color.b * cmd->data.color.color_scale, 0.0f, 1.0f) * 255.0f);
  953. const Uint8 a = (Uint8)SDL_roundf(SDL_clamp(cmd->data.color.color.a, 0.0f, 1.0f) * 255.0f);
  954. sceGuClearColor(GU_RGBA(r, g, b, a));
  955. sceGuClearStencil(a);
  956. sceGuClear(GU_COLOR_BUFFER_BIT | GU_STENCIL_BUFFER_BIT);
  957. break;
  958. }
  959. case SDL_RENDERCMD_DRAW_POINTS:
  960. {
  961. const size_t count = cmd->data.draw.count;
  962. const VertV *verts = (VertV *)(gpumem + cmd->data.draw.first);
  963. PSP_BlendState state = {
  964. .color = drawstate.color,
  965. .texture = NULL,
  966. .texture_scale_mode = SDL_SCALEMODE_INVALID,
  967. .texture_address_mode = SDL_TEXTURE_ADDRESS_INVALID,
  968. .mode = cmd->data.draw.blend,
  969. .shadeModel = GU_FLAT
  970. };
  971. PSP_SetBlendState(data, &state);
  972. sceGuDrawArray(GU_POINTS, GU_VERTEX_32BITF | GU_TRANSFORM_2D, count, 0, verts);
  973. break;
  974. }
  975. case SDL_RENDERCMD_DRAW_LINES:
  976. {
  977. const size_t count = cmd->data.draw.count;
  978. const VertV *verts = (VertV *)(gpumem + cmd->data.draw.first);
  979. PSP_BlendState state = {
  980. .color = drawstate.color,
  981. .texture = NULL,
  982. .texture_scale_mode = SDL_SCALEMODE_INVALID,
  983. .texture_address_mode = SDL_TEXTURE_ADDRESS_INVALID,
  984. .mode = cmd->data.draw.blend,
  985. .shadeModel = GU_FLAT
  986. };
  987. PSP_SetBlendState(data, &state);
  988. sceGuDrawArray(GU_LINE_STRIP, GU_VERTEX_32BITF | GU_TRANSFORM_2D, count, 0, verts);
  989. break;
  990. }
  991. case SDL_RENDERCMD_FILL_RECTS:
  992. {
  993. const size_t count = cmd->data.draw.count;
  994. const VertV *verts = (VertV *)(gpumem + cmd->data.draw.first);
  995. PSP_BlendState state = {
  996. .color = drawstate.color,
  997. .texture = NULL,
  998. .texture_scale_mode = SDL_SCALEMODE_INVALID,
  999. .texture_address_mode = SDL_TEXTURE_ADDRESS_INVALID,
  1000. .mode = cmd->data.draw.blend,
  1001. .shadeModel = GU_FLAT
  1002. };
  1003. PSP_SetBlendState(data, &state);
  1004. sceGuDrawArray(GU_SPRITES, GU_VERTEX_32BITF | GU_TRANSFORM_2D, 2 * count, 0, verts);
  1005. break;
  1006. }
  1007. case SDL_RENDERCMD_COPY:
  1008. {
  1009. const size_t count = cmd->data.draw.count;
  1010. const VertTV *verts = (VertTV *)(gpumem + cmd->data.draw.first);
  1011. PSP_BlendState state = {
  1012. .color = drawstate.color,
  1013. .texture = cmd->data.draw.texture,
  1014. .texture_scale_mode = cmd->data.draw.texture_scale_mode,
  1015. .texture_address_mode = cmd->data.draw.texture_address_mode,
  1016. .mode = cmd->data.draw.blend,
  1017. .shadeModel = GU_SMOOTH
  1018. };
  1019. PSP_SetBlendState(data, &state);
  1020. sceGuDrawArray(GU_SPRITES, GU_TEXTURE_32BITF | GU_VERTEX_32BITF | GU_TRANSFORM_2D, 2 * count, 0, verts);
  1021. break;
  1022. }
  1023. case SDL_RENDERCMD_COPY_EX:
  1024. {
  1025. const VertTV *verts = (VertTV *)(gpumem + cmd->data.draw.first);
  1026. PSP_BlendState state = {
  1027. .color = drawstate.color,
  1028. .texture = cmd->data.draw.texture,
  1029. .texture_scale_mode = cmd->data.draw.texture_scale_mode,
  1030. .texture_address_mode = cmd->data.draw.texture_address_mode,
  1031. .mode = cmd->data.draw.blend,
  1032. .shadeModel = GU_SMOOTH
  1033. };
  1034. PSP_SetBlendState(data, &state);
  1035. sceGuDrawArray(GU_TRIANGLE_FAN, GU_TEXTURE_32BITF | GU_VERTEX_32BITF | GU_TRANSFORM_2D, 4, 0, verts);
  1036. break;
  1037. }
  1038. case SDL_RENDERCMD_GEOMETRY:
  1039. {
  1040. const size_t count = cmd->data.draw.count;
  1041. if (!cmd->data.draw.texture) {
  1042. const VertCV *verts = (VertCV *)(gpumem + cmd->data.draw.first);
  1043. sceGuDisable(GU_TEXTURE_2D);
  1044. // In GU_SMOOTH mode
  1045. sceGuDrawArray(GU_TRIANGLES, GU_COLOR_8888 | GU_VERTEX_32BITF | GU_TRANSFORM_2D, count, 0, verts);
  1046. sceGuEnable(GU_TEXTURE_2D);
  1047. } else {
  1048. const VertTCV *verts = (VertTCV *)(gpumem + cmd->data.draw.first);
  1049. PSP_BlendState state = {
  1050. .color = drawstate.color,
  1051. .texture = cmd->data.draw.texture,
  1052. .texture_scale_mode = cmd->data.draw.texture_scale_mode,
  1053. .texture_address_mode = cmd->data.draw.texture_address_mode,
  1054. .mode = cmd->data.draw.blend,
  1055. .shadeModel = GU_SMOOTH
  1056. };
  1057. PSP_SetBlendState(data, &state);
  1058. sceGuDrawArray(GU_TRIANGLES, GU_TEXTURE_32BITF | GU_COLOR_8888 | GU_VERTEX_32BITF | GU_TRANSFORM_2D, count, 0, verts);
  1059. }
  1060. break;
  1061. }
  1062. case SDL_RENDERCMD_NO_OP:
  1063. break;
  1064. }
  1065. cmd = cmd->next;
  1066. }
  1067. return true;
  1068. }
  1069. static bool PSP_RenderPresent(SDL_Renderer *renderer)
  1070. {
  1071. PSP_RenderData *data = (PSP_RenderData *)renderer->internal;
  1072. if (!data->displayListAvail) {
  1073. return false;
  1074. }
  1075. data->displayListAvail = false;
  1076. sceGuFinish();
  1077. sceGuSync(0, 0);
  1078. if ((data->vsync) && (data->vblank_not_reached)) {
  1079. sceDisplayWaitVblankStart();
  1080. }
  1081. data->vblank_not_reached = true;
  1082. data->backbuffer = data->frontbuffer;
  1083. data->frontbuffer = vabsptr(sceGuSwapBuffers());
  1084. return true;
  1085. }
  1086. static void PSP_DestroyTexture(SDL_Renderer *renderer, SDL_Texture *texture)
  1087. {
  1088. PSP_RenderData *renderdata = (PSP_RenderData *)renderer->internal;
  1089. PSP_TextureData *psp_texture = (PSP_TextureData *)texture->internal;
  1090. if (!renderdata) {
  1091. return;
  1092. }
  1093. if (!psp_texture) {
  1094. return;
  1095. }
  1096. LRUTargetRemove(renderdata, psp_texture);
  1097. TextureStorageFree(psp_texture->data);
  1098. SDL_free(psp_texture);
  1099. texture->internal = NULL;
  1100. }
  1101. static void PSP_DestroyRenderer(SDL_Renderer *renderer)
  1102. {
  1103. PSP_RenderData *data = (PSP_RenderData *)renderer->internal;
  1104. if (data) {
  1105. if (!data->initialized) {
  1106. return;
  1107. }
  1108. sceKernelDisableSubIntr(PSP_VBLANK_INT, 0);
  1109. sceKernelReleaseSubIntrHandler(PSP_VBLANK_INT, 0);
  1110. sceDisplayWaitVblankStart();
  1111. sceGuDisplay(GU_FALSE);
  1112. sceGuTerm();
  1113. vfree(data->backbuffer);
  1114. vfree(data->frontbuffer);
  1115. data->initialized = false;
  1116. data->displayListAvail = false;
  1117. SDL_free(data);
  1118. }
  1119. }
  1120. static bool PSP_SetVSync(SDL_Renderer *renderer, const int vsync)
  1121. {
  1122. PSP_RenderData *data = renderer->internal;
  1123. data->vsync = vsync;
  1124. return true;
  1125. }
  1126. static bool PSP_CreateRenderer(SDL_Renderer *renderer, SDL_Window *window, SDL_PropertiesID create_props)
  1127. {
  1128. PSP_RenderData *data;
  1129. int pixelformat;
  1130. void *doublebuffer = NULL;
  1131. SDL_SetupRendererColorspace(renderer, create_props);
  1132. if (renderer->output_colorspace != SDL_COLORSPACE_SRGB) {
  1133. return SDL_SetError("Unsupported output colorspace");
  1134. }
  1135. data = (PSP_RenderData *)SDL_calloc(1, sizeof(*data));
  1136. if (!data) {
  1137. return false;
  1138. }
  1139. renderer->WindowEvent = PSP_WindowEvent;
  1140. renderer->CreateTexture = PSP_CreateTexture;
  1141. renderer->UpdateTexture = PSP_UpdateTexture;
  1142. renderer->LockTexture = PSP_LockTexture;
  1143. renderer->UnlockTexture = PSP_UnlockTexture;
  1144. renderer->SetRenderTarget = PSP_SetRenderTarget;
  1145. renderer->QueueSetViewport = PSP_QueueNoOp;
  1146. renderer->QueueSetDrawColor = PSP_QueueNoOp;
  1147. renderer->QueueDrawPoints = PSP_QueueDrawPoints;
  1148. renderer->QueueDrawLines = PSP_QueueDrawPoints; // lines and points queue vertices the same way.
  1149. renderer->QueueGeometry = PSP_QueueGeometry;
  1150. renderer->QueueFillRects = PSP_QueueFillRects;
  1151. renderer->QueueCopy = PSP_QueueCopy;
  1152. renderer->QueueCopyEx = PSP_QueueCopyEx;
  1153. renderer->InvalidateCachedState = PSP_InvalidateCachedState;
  1154. renderer->RunCommandQueue = PSP_RunCommandQueue;
  1155. renderer->RenderPresent = PSP_RenderPresent;
  1156. renderer->DestroyTexture = PSP_DestroyTexture;
  1157. renderer->DestroyRenderer = PSP_DestroyRenderer;
  1158. renderer->SetVSync = PSP_SetVSync;
  1159. renderer->internal = data;
  1160. PSP_InvalidateCachedState(renderer);
  1161. renderer->window = window;
  1162. renderer->name = PSP_RenderDriver.name;
  1163. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_BGR565);
  1164. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_ABGR1555);
  1165. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_ABGR4444);
  1166. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_ABGR8888);
  1167. SDL_SetNumberProperty(SDL_GetRendererProperties(renderer), SDL_PROP_RENDERER_MAX_TEXTURE_SIZE_NUMBER, 512);
  1168. data->initialized = true;
  1169. data->most_recent_target = NULL;
  1170. data->least_recent_target = NULL;
  1171. pixelformat = PixelFormatToPSPFMT(SDL_GetWindowPixelFormat(window));
  1172. switch (pixelformat) {
  1173. case GU_PSM_4444:
  1174. case GU_PSM_5650:
  1175. case GU_PSM_5551:
  1176. data->bpp = 2;
  1177. data->psm = pixelformat;
  1178. break;
  1179. default:
  1180. data->bpp = 4;
  1181. data->psm = GU_PSM_8888;
  1182. break;
  1183. }
  1184. doublebuffer = vramalloc(PSP_FRAME_BUFFER_SIZE * data->bpp * 2);
  1185. data->backbuffer = doublebuffer;
  1186. data->frontbuffer = ((uint8_t *)doublebuffer) + PSP_FRAME_BUFFER_SIZE * data->bpp;
  1187. sceGuInit();
  1188. // setup GU
  1189. sceGuStart(GU_DIRECT, DisplayList);
  1190. sceGuDrawBuffer(data->psm, vrelptr(data->frontbuffer), PSP_FRAME_BUFFER_WIDTH);
  1191. sceGuDispBuffer(PSP_SCREEN_WIDTH, PSP_SCREEN_HEIGHT, vrelptr(data->backbuffer), PSP_FRAME_BUFFER_WIDTH);
  1192. sceGuOffset(2048 - (PSP_SCREEN_WIDTH >> 1), 2048 - (PSP_SCREEN_HEIGHT >> 1));
  1193. sceGuViewport(2048, 2048, PSP_SCREEN_WIDTH, PSP_SCREEN_HEIGHT);
  1194. sceGuDisable(GU_DEPTH_TEST);
  1195. // Scissoring
  1196. sceGuScissor(0, 0, PSP_SCREEN_WIDTH, PSP_SCREEN_HEIGHT);
  1197. sceGuEnable(GU_SCISSOR_TEST);
  1198. // Backface culling
  1199. sceGuDisable(GU_CULL_FACE);
  1200. // Setup initial blend state
  1201. ResetBlendState(&data->blendState);
  1202. sceGuFinish();
  1203. sceGuSync(0, 0);
  1204. sceDisplayWaitVblankStartCB();
  1205. sceGuDisplay(GU_TRUE);
  1206. // Improve performance when VSYC is enabled and it is not reaching the 60 FPS
  1207. data->vblank_not_reached = true;
  1208. sceKernelRegisterSubIntrHandler(PSP_VBLANK_INT, 0, psp_on_vblank, data);
  1209. sceKernelEnableSubIntr(PSP_VBLANK_INT, 0);
  1210. return true;
  1211. }
  1212. SDL_RenderDriver PSP_RenderDriver = {
  1213. PSP_CreateRenderer, "PSP"
  1214. };
  1215. #endif // SDL_VIDEO_RENDER_PSP