/modules/iopcore/cdvdman/dev9.c

https://bitbucket.org/ifcaro/open-ps2-loader/ · C · 640 lines · 480 code · 115 blank · 45 comment · 103 complexity · 11d9370657fa105381a95c744c4a284d MD5 · raw file

  1. /*
  2. # _____ ___ ____ ___ ____
  3. # ____| | ____| | | |____|
  4. # | ___| |____ ___| ____| | \ PS2DEV Open Source Project.
  5. #-----------------------------------------------------------------------
  6. # Copyright (c) 2003 Marcus R. Brown <mrbrown@0xd6.org>
  7. # Licenced under Academic Free License version 2.0
  8. # Review ps2sdk README & LICENSE files for further details.
  9. #
  10. # $Id: ps2dev9.c 1454 2007-11-04 23:19:57Z roman_ps2dev $
  11. # DEV9 Device Driver.
  12. */
  13. #include <types.h>
  14. #include <defs.h>
  15. #include <loadcore.h>
  16. #include <intrman.h>
  17. #include <dmacman.h>
  18. #include <thbase.h>
  19. #include <thsemap.h>
  20. #include <aifregs.h>
  21. #include <dev9regs.h>
  22. #include <speedregs.h>
  23. #include <smapregs.h>
  24. #include <stdio.h>
  25. #include "dev9.h"
  26. #include "ioman.h"
  27. #include "ioman_add.h"
  28. #ifdef HDD_DRIVER
  29. extern char atad_inited;
  30. #endif
  31. #ifdef DEV9_DEBUG
  32. #define M_PRINTF(format, args...) \
  33. printf(MODNAME ": " format, ## args)
  34. #else
  35. #define M_PRINTF(format, args...) \
  36. do {} while (0)
  37. #endif
  38. #define VERSION "1.0"
  39. #define BANNER "\nDEV9 device driver v%s - Copyright (c) 2003 Marcus R. Brown\n\n"
  40. #define DEV9_INTR 13
  41. /* SSBUS registers. */
  42. #define SSBUS_R_1418 0xbf801418
  43. #define SSBUS_R_141c 0xbf80141c
  44. #define SSBUS_R_1420 0xbf801420
  45. static int dev9type = -1; /* 0 for PCMCIA, 1 for expansion bay */
  46. static int using_aif = 0; /* 1 if using AIF on a T10K */
  47. static void (*p_dev9_intr_cb)(int flag) = NULL;
  48. static int dma_lock_sem = -1; /* used to arbitrate DMA */
  49. static s16 eeprom_data[5]; /* 2-byte EEPROM status (0/-1 = invalid, 1 = valid),
  50. 6-byte MAC address,
  51. 2-byte MAC address checksum. */
  52. /* Each driver can register callbacks that correspond to each bit of the
  53. SMAP interrupt status register (0xbx000028). */
  54. static dev9_intr_cb_t dev9_intr_cbs[16];
  55. static dev9_shutdown_cb_t dev9_shutdown_cbs[16];
  56. static dev9_dma_cb_t dev9_predma_cbs[4], dev9_postdma_cbs[4];
  57. static int dev9_intr_dispatch(int flag);
  58. static void dev9_set_stat(int stat);
  59. static int read_eeprom_data(void);
  60. static int dev9_init(void);
  61. static void pcmcia_set_stat(int stat);
  62. #ifdef DEBUG
  63. static int pcmcia_device_probe(void);
  64. #endif
  65. static int pcmcia_init(void);
  66. static void expbay_set_stat(int stat);
  67. static int expbay_init(void);
  68. static int dev9x_dummy(void) { return 0; }
  69. static int dev9x_devctl(iop_file_t *f, const char *name, int cmd, void *args, int arglen, void *buf, int buflen)
  70. {
  71. if (cmd == 0x4401)
  72. return dev9type;
  73. return 0;
  74. }
  75. /* driver ops func tab */
  76. static void *dev9x_ops[27] = {
  77. (void*)dev9x_dummy,
  78. (void*)dev9x_dummy,
  79. (void*)dev9x_dummy,
  80. (void*)dev9x_dummy,
  81. (void*)dev9x_dummy,
  82. (void*)dev9x_dummy,
  83. (void*)dev9x_dummy,
  84. (void*)dev9x_dummy,
  85. (void*)dev9x_dummy,
  86. (void*)dev9x_dummy,
  87. (void*)dev9x_dummy,
  88. (void*)dev9x_dummy,
  89. (void*)dev9x_dummy,
  90. (void*)dev9x_dummy,
  91. (void*)dev9x_dummy,
  92. (void*)dev9x_dummy,
  93. (void*)dev9x_dummy,
  94. (void*)dev9x_dummy,
  95. (void*)dev9x_dummy,
  96. (void*)dev9x_dummy,
  97. (void*)dev9x_dummy,
  98. (void*)dev9x_dummy,
  99. (void*)dev9x_dummy,
  100. (void*)dev9x_devctl,
  101. (void*)dev9_init,
  102. (void*)dev9_init,
  103. (void*)dev9_init
  104. };
  105. /* driver descriptor */
  106. static iop_ext_device_t dev9x_dev = {
  107. "dev9x",
  108. IOP_DT_FS | 0x10000000, /* EXT FS */
  109. 1,
  110. "DEV9",
  111. (struct _iop_ext_device_ops *)&dev9x_ops
  112. };
  113. int dev9d_init(void)
  114. {
  115. USE_DEV9_REGS;
  116. int idx, res = 1;
  117. u16 dev9hw;
  118. M_PRINTF(BANNER, VERSION);
  119. for (idx = 0; idx < 16; idx++)
  120. dev9_shutdown_cbs[idx] = NULL;
  121. dev9hw = DEV9_REG(DEV9_R_REV) & 0xf0;
  122. if (dev9hw == 0x20) { /* CXD9566 (PCMCIA) */
  123. dev9type = 0;
  124. res = pcmcia_init();
  125. } else if (dev9hw == 0x30) { /* CXD9611 (Expansion Bay) */
  126. dev9type = 1;
  127. res = expbay_init();
  128. }
  129. if (res)
  130. return res;
  131. DelDrv("dev9x");
  132. AddDrv((iop_device_t *)&dev9x_dev);
  133. return 0;
  134. }
  135. /* Export 4 */
  136. void dev9RegisterIntrCb(int intr, dev9_intr_cb_t cb)
  137. {
  138. #ifdef HDD_DRIVER
  139. //Don't let anything else change the HDD interrupt handlers.
  140. if (intr < 2) {
  141. if (atad_inited)
  142. return;
  143. }
  144. #endif
  145. dev9_intr_cbs[intr] = cb;
  146. }
  147. /* Export 12 */
  148. void dev9RegisterPreDmaCb(int ctrl, dev9_dma_cb_t cb){
  149. dev9_predma_cbs[ctrl] = cb;
  150. }
  151. /* Export 13 */
  152. void dev9RegisterPostDmaCb(int ctrl, dev9_dma_cb_t cb){
  153. dev9_postdma_cbs[ctrl] = cb;
  154. }
  155. /* flag is 1 if a card (pcmcia) was removed or added */
  156. static int dev9_intr_dispatch(int flag)
  157. {
  158. USE_SPD_REGS;
  159. int i, bit;
  160. if (flag) {
  161. for (i = 0; i < 16; i++)
  162. if (dev9_intr_cbs[i] != NULL)
  163. dev9_intr_cbs[i](flag);
  164. }
  165. while (SPD_REG16(SPD_R_INTR_STAT) & SPD_REG16(SPD_R_INTR_MASK)) {
  166. for (i = 0; i < 16; i++) {
  167. if (dev9_intr_cbs[i] != NULL) {
  168. bit = (SPD_REG16(SPD_R_INTR_STAT) &
  169. SPD_REG16(SPD_R_INTR_MASK)) >> i;
  170. if (bit & 0x01)
  171. dev9_intr_cbs[i](flag);
  172. }
  173. }
  174. }
  175. return 0;
  176. }
  177. static void dev9_set_stat(int stat)
  178. {
  179. if (dev9type == 0)
  180. pcmcia_set_stat(stat);
  181. else if (dev9type == 1)
  182. expbay_set_stat(stat);
  183. }
  184. /* Export 6 */
  185. void dev9Shutdown(void)
  186. {
  187. int idx;
  188. USE_DEV9_REGS;
  189. for (idx = 0; idx < 16; idx++)
  190. if (dev9_shutdown_cbs[idx])
  191. dev9_shutdown_cbs[idx]();
  192. if (dev9type == 0) { /* PCMCIA */
  193. DEV9_REG(DEV9_R_POWER) = 0;
  194. DEV9_REG(DEV9_R_1474) = 0;
  195. } else if (dev9type == 1) {
  196. DEV9_REG(DEV9_R_1466) = 1;
  197. DEV9_REG(DEV9_R_1464) = 0;
  198. DEV9_REG(DEV9_R_1460) = DEV9_REG(DEV9_R_1464);
  199. DEV9_REG(DEV9_R_POWER) = DEV9_REG(DEV9_R_POWER) & ~4;
  200. DEV9_REG(DEV9_R_POWER) = DEV9_REG(DEV9_R_POWER) & ~1;
  201. }
  202. DelayThread(1000000);
  203. }
  204. /* Export 7 */
  205. void dev9IntrEnable(int mask)
  206. {
  207. USE_SPD_REGS;
  208. int flags;
  209. CpuSuspendIntr(&flags);
  210. SPD_REG16(SPD_R_INTR_MASK) = SPD_REG16(SPD_R_INTR_MASK) | mask;
  211. CpuResumeIntr(flags);
  212. }
  213. /* Export 8 */
  214. void dev9IntrDisable(int mask)
  215. {
  216. USE_SPD_REGS;
  217. int flags;
  218. CpuSuspendIntr(&flags);
  219. SPD_REG16(SPD_R_INTR_MASK) = SPD_REG16(SPD_R_INTR_MASK) & ~mask;
  220. CpuResumeIntr(flags);
  221. }
  222. /* Export 5 */
  223. int dev9DmaTransfer(int ctrl, void *buf, int bcr, int dir)
  224. {
  225. USE_SPD_REGS;
  226. int res = 0, dmactrl;
  227. switch(ctrl){
  228. case 0:
  229. case 1: dmactrl = ctrl;
  230. break;
  231. case 2:
  232. case 3:
  233. if (dev9_predma_cbs[ctrl] == NULL) return -1;
  234. if (dev9_postdma_cbs[ctrl] == NULL) return -1;
  235. dmactrl = (4 << ctrl);
  236. break;
  237. default:
  238. return -1;
  239. }
  240. if ((res = WaitSema(dma_lock_sem)) < 0)
  241. return res;
  242. SPD_REG16(SPD_R_DMA_CTRL) = (SPD_REG16(SPD_R_REV_1)<17)?(dmactrl&0x03)|0x04:(dmactrl&0x01)|0x06;
  243. if (dev9_predma_cbs[ctrl])
  244. dev9_predma_cbs[ctrl](bcr, dir);
  245. dmac_request(IOP_DMAC_DEV9, buf, bcr&0xFFFF, bcr>>16, dir);
  246. dmac_transfer(IOP_DMAC_DEV9);
  247. /* Wait for DMA to complete. Do not use a semaphore as thread switching hurts throughput greatly. */
  248. while(dmac_ch_get_chcr(IOP_DMAC_DEV9)&DMAC_CHCR_TR){}
  249. res=0;
  250. if (dev9_postdma_cbs[ctrl])
  251. dev9_postdma_cbs[ctrl](bcr, dir);
  252. SignalSema(dma_lock_sem);
  253. return res;
  254. }
  255. static int read_eeprom_data(void)
  256. {
  257. USE_SPD_REGS;
  258. int i, j, res = -2;
  259. u8 val;
  260. if (eeprom_data[0] < 0)
  261. goto out;
  262. SPD_REG8(SPD_R_PIO_DIR) = 0xe1;
  263. DelayThread(1);
  264. SPD_REG8(SPD_R_PIO_DATA) = 0x80;
  265. DelayThread(1);
  266. for (i = 0; i < 2; i++) {
  267. SPD_REG8(SPD_R_PIO_DATA) = 0xa0;
  268. DelayThread(1);
  269. SPD_REG8(SPD_R_PIO_DATA) = 0xe0;
  270. DelayThread(1);
  271. }
  272. for (i = 0; i < 7; i++) {
  273. SPD_REG8(SPD_R_PIO_DATA) = 0x80;
  274. DelayThread(1);
  275. SPD_REG8(SPD_R_PIO_DATA) = 0xc0;
  276. DelayThread(1);
  277. }
  278. SPD_REG8(SPD_R_PIO_DATA) = 0xc0;
  279. DelayThread(1);
  280. val = SPD_REG8(SPD_R_PIO_DATA);
  281. DelayThread(1);
  282. if (val & 0x10) { /* Error. */
  283. SPD_REG8(SPD_R_PIO_DATA) = 0;
  284. DelayThread(1);
  285. res = -1;
  286. eeprom_data[0] = 0;
  287. goto out;
  288. }
  289. SPD_REG8(SPD_R_PIO_DATA) = 0x80;
  290. DelayThread(1);
  291. /* Read the MAC address and checksum from the EEPROM. */
  292. for (i = 0; i < 4; i++) {
  293. eeprom_data[i+1] = 0;
  294. for (j = 15; j >= 0; j--) {
  295. SPD_REG8(SPD_R_PIO_DATA) = 0xc0;
  296. DelayThread(1);
  297. val = SPD_REG8(SPD_R_PIO_DATA);
  298. if (val & 0x10)
  299. eeprom_data[i+1] |= (1<<j);
  300. SPD_REG8(SPD_R_PIO_DATA) = 0x80;
  301. DelayThread(1);
  302. }
  303. }
  304. SPD_REG8(SPD_R_PIO_DATA) = 0;
  305. DelayThread(1);
  306. eeprom_data[0] = 1; /* The EEPROM data is valid. */
  307. res = 0;
  308. out:
  309. SPD_REG8(SPD_R_PIO_DIR) = 1;
  310. return res;
  311. }
  312. /* Export 9 */
  313. int dev9GetEEPROM(u16 *buf)
  314. {
  315. int i;
  316. if (!eeprom_data[0])
  317. return -1;
  318. if (eeprom_data[0] < 0)
  319. return -2;
  320. /* We only return the MAC address and checksum. */
  321. for (i = 0; i < 4; i++)
  322. buf[i] = eeprom_data[i+1];
  323. return 0;
  324. }
  325. /* Export 10 */
  326. void dev9LEDCtl(int ctl)
  327. {
  328. USE_SPD_REGS;
  329. SPD_REG8(SPD_R_PIO_DATA) = (ctl == 0);
  330. }
  331. /* Export 11 */
  332. int dev9RegisterShutdownCb(int idx, dev9_shutdown_cb_t cb){
  333. if (idx < 16)
  334. {
  335. dev9_shutdown_cbs[idx] = cb;
  336. return 0;
  337. }
  338. return -1;
  339. }
  340. static int dev9_init(void)
  341. {
  342. int i;
  343. if ((dma_lock_sem = CreateMutex(IOP_MUTEX_UNLOCKED)) < 0)
  344. return -1;
  345. /* Enable the DEV9 DMAC channel. */
  346. dmac_enable(IOP_DMAC_DEV9);
  347. /* Not quite sure what this enables yet. */
  348. dev9_set_stat(0x103);
  349. /* Disable all device interrupts. */
  350. dev9IntrDisable(0xffff);
  351. p_dev9_intr_cb = (void *)dev9_intr_dispatch;
  352. /* Reset the SMAP interrupt callback table. */
  353. for (i = 0; i < 16; i++)
  354. dev9_intr_cbs[i] = NULL;
  355. for (i = 0; i < 4; i++){
  356. dev9_predma_cbs[i] = NULL;
  357. dev9_postdma_cbs[i] = NULL;
  358. }
  359. /* Read in the MAC address. */
  360. read_eeprom_data();
  361. /* Turn the LED off. */
  362. dev9LEDCtl(0);
  363. return 0;
  364. }
  365. #ifdef DEBUG
  366. static int pcic_get_cardtype(void)
  367. {
  368. USE_DEV9_REGS;
  369. u16 val = DEV9_REG(DEV9_R_1462) & 0x03;
  370. if (val == 0)
  371. return 1; /* 16-bit */
  372. else
  373. if (val < 3)
  374. return 2; /* CardBus */
  375. return 0;
  376. }
  377. static int pcic_get_voltage(void)
  378. {
  379. USE_DEV9_REGS;
  380. u16 val = DEV9_REG(DEV9_R_1462) & 0x0c;
  381. if (val == 0x04)
  382. return 3;
  383. if (val == 0 || val == 0x08)
  384. return 1;
  385. if (val == 0x0c)
  386. return 2;
  387. return 0;
  388. }
  389. #endif
  390. static void pcmcia_set_stat(int stat)
  391. {
  392. USE_DEV9_REGS;
  393. u16 val = stat & 0x01;
  394. if (stat & 0x10)
  395. val = 1;
  396. if (stat & 0x02)
  397. val |= 0x02;
  398. if (stat & 0x20)
  399. val |= 0x02;
  400. if (stat & 0x04)
  401. val |= 0x08;
  402. if (stat & 0x08)
  403. val |= 0x10;
  404. if (stat & 0x200)
  405. val |= 0x20;
  406. if (stat & 0x100)
  407. val |= 0x40;
  408. if (stat & 0x400)
  409. val |= 0x80;
  410. if (stat & 0x800)
  411. val |= 0x04;
  412. DEV9_REG(DEV9_R_1476) = val & 0xff;
  413. }
  414. #ifdef DEBUG
  415. static int pcmcia_device_probe(void)
  416. {
  417. const char *pcic_ct_names[] = { "No", "16-bit", "CardBus" };
  418. int voltage;
  419. int pcic_cardtype; /* Translated value of bits 0-1 of 0xbf801462 */
  420. int pcic_voltage; /* Translated value of bits 2-3 of 0xbf801462 */
  421. pcic_voltage = pcic_get_voltage();
  422. pcic_cardtype = pcic_get_cardtype();
  423. voltage = (pcic_voltage == 2 ? 5 : (pcic_voltage == 1 ? 3 : 0));
  424. M_PRINTF("%s PCMCIA card detected. Vcc = %dV\n",
  425. pcic_ct_names[pcic_cardtype], voltage);
  426. if (pcic_voltage == 3 || pcic_cardtype != 1)
  427. return -1;
  428. return 0;
  429. }
  430. #endif
  431. static int pcmcia_intr(void *unused)
  432. {
  433. USE_AIF_REGS;
  434. USE_DEV9_REGS;
  435. u16 cstc1, cstc2;
  436. cstc1 = DEV9_REG(DEV9_R_1464);
  437. cstc2 = DEV9_REG(DEV9_R_1466);
  438. if (using_aif) {
  439. if (aif_regs[AIF_INTSR] & AIF_INTR_PCMCIA)
  440. aif_regs[AIF_INTCL] = AIF_INTR_PCMCIA;
  441. else
  442. return 0; /* Unknown interrupt. */
  443. }
  444. /* Acknowledge the interrupt. */
  445. DEV9_REG(DEV9_R_1464) = cstc1;
  446. DEV9_REG(DEV9_R_1466) = cstc2;
  447. if (cstc1 & 0x03 || cstc2 & 0x03) { /* Card removed or added? */
  448. if (p_dev9_intr_cb)
  449. p_dev9_intr_cb(1);
  450. /* Shutdown the card. */
  451. DEV9_REG(DEV9_R_POWER) = 0;
  452. DEV9_REG(DEV9_R_1474) = 0;
  453. #ifdef DEBUG
  454. pcmcia_device_probe();
  455. #endif
  456. }
  457. if (cstc1 & 0x80 || cstc2 & 0x80) {
  458. if (p_dev9_intr_cb)
  459. p_dev9_intr_cb(0);
  460. }
  461. DEV9_REG(DEV9_R_147E) = 1;
  462. DEV9_REG(DEV9_R_147E) = 0;
  463. return 1;
  464. }
  465. static int pcmcia_init(void)
  466. {
  467. USE_DEV9_REGS;
  468. USE_AIF_REGS;
  469. int *mode;
  470. _sw(0x51011, SSBUS_R_1420);
  471. _sw(0x1a00bb, SSBUS_R_1418);
  472. _sw(0xef1a3043, SSBUS_R_141c);
  473. /* If we are a T10K, then we go through AIF. */
  474. if ((mode = QueryBootMode(6)) != NULL) {
  475. if ((*(u16 *)mode & 0xfe) == 0x60) {
  476. M_PRINTF("T10K detected.\n");
  477. if (aif_regs[AIF_IDENT] == 0xa1) {
  478. aif_regs[AIF_INTEN] = AIF_INTR_PCMCIA;
  479. using_aif = 1;
  480. } else {
  481. M_PRINTF("AIF not detected.\n");
  482. return 1;
  483. }
  484. }
  485. }
  486. //The DEV9 interface and SPEED should be already initialized.
  487. _sw(0xe01a3043, SSBUS_R_1418);
  488. if (dev9_init() != 0)
  489. return 1;
  490. RegisterIntrHandler(IOP_IRQ_DEV9, 1, &pcmcia_intr, NULL);
  491. EnableIntr(IOP_IRQ_DEV9);
  492. DEV9_REG(DEV9_R_147E) = 0;
  493. M_PRINTF("CXD9566 (PCMCIA type) initialized.\n");
  494. return 0;
  495. }
  496. static void expbay_set_stat(int stat)
  497. {
  498. USE_DEV9_REGS;
  499. DEV9_REG(DEV9_R_1464) = stat & 0x3f;
  500. }
  501. static int expbay_intr(void *unused)
  502. {
  503. USE_DEV9_REGS;
  504. if (p_dev9_intr_cb)
  505. p_dev9_intr_cb(0);
  506. DEV9_REG(DEV9_R_1466) = 1;
  507. DEV9_REG(DEV9_R_1466) = 0;
  508. return 1;
  509. }
  510. static int expbay_init(void)
  511. {
  512. USE_DEV9_REGS;
  513. _sw(0x51011, SSBUS_R_1420);
  514. _sw(0xe01a3043, SSBUS_R_1418);
  515. _sw(0xef1a3043, SSBUS_R_141c);
  516. //The DEV9 interface and SPEED should be already initialized.
  517. if (dev9_init() != 0)
  518. return 1;
  519. ReleaseIntrHandler(IOP_IRQ_DEV9);
  520. RegisterIntrHandler(IOP_IRQ_DEV9, 1, expbay_intr, NULL);
  521. EnableIntr(IOP_IRQ_DEV9);
  522. DEV9_REG(DEV9_R_1466) = 0;
  523. M_PRINTF("CXD9611 (Expansion Bay type) initialized.\n");
  524. return 0;
  525. }