pcmcia.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725
  1. /*
  2. * Sonics Silicon Backplane
  3. * PCMCIA-Hostbus related functions
  4. *
  5. * Copyright 2006 Johannes Berg <johannes@sipsolutions.net>
  6. * Copyright 2007-2008 Michael Buesch <mb@bu3sch.de>
  7. *
  8. * Licensed under the GNU/GPL. See COPYING for details.
  9. */
  10. #include <linux/ssb/ssb.h>
  11. #include <linux/delay.h>
  12. #include <linux/io.h>
  13. #include <linux/etherdevice.h>
  14. #include <pcmcia/cs_types.h>
  15. #include <pcmcia/cs.h>
  16. #include <pcmcia/cistpl.h>
  17. #include <pcmcia/ciscode.h>
  18. #include <pcmcia/ds.h>
  19. #include <pcmcia/cisreg.h>
  20. #include "ssb_private.h"
  21. /* Define the following to 1 to enable a printk on each coreswitch. */
  22. #define SSB_VERBOSE_PCMCIACORESWITCH_DEBUG 0
  23. /* PCMCIA configuration registers */
  24. #define SSB_PCMCIA_ADDRESS0 0x2E
  25. #define SSB_PCMCIA_ADDRESS1 0x30
  26. #define SSB_PCMCIA_ADDRESS2 0x32
  27. #define SSB_PCMCIA_MEMSEG 0x34
  28. #define SSB_PCMCIA_SPROMCTL 0x36
  29. #define SSB_PCMCIA_SPROMCTL_IDLE 0
  30. #define SSB_PCMCIA_SPROMCTL_WRITE 1
  31. #define SSB_PCMCIA_SPROMCTL_READ 2
  32. #define SSB_PCMCIA_SPROMCTL_WRITEEN 4
  33. #define SSB_PCMCIA_SPROMCTL_WRITEDIS 7
  34. #define SSB_PCMCIA_SPROMCTL_DONE 8
  35. #define SSB_PCMCIA_SPROM_DATALO 0x38
  36. #define SSB_PCMCIA_SPROM_DATAHI 0x3A
  37. #define SSB_PCMCIA_SPROM_ADDRLO 0x3C
  38. #define SSB_PCMCIA_SPROM_ADDRHI 0x3E
  39. /* Hardware invariants CIS tuples */
  40. #define SSB_PCMCIA_CIS 0x80
  41. #define SSB_PCMCIA_CIS_ID 0x01
  42. #define SSB_PCMCIA_CIS_BOARDREV 0x02
  43. #define SSB_PCMCIA_CIS_PA 0x03
  44. #define SSB_PCMCIA_CIS_PA_PA0B0_LO 0
  45. #define SSB_PCMCIA_CIS_PA_PA0B0_HI 1
  46. #define SSB_PCMCIA_CIS_PA_PA0B1_LO 2
  47. #define SSB_PCMCIA_CIS_PA_PA0B1_HI 3
  48. #define SSB_PCMCIA_CIS_PA_PA0B2_LO 4
  49. #define SSB_PCMCIA_CIS_PA_PA0B2_HI 5
  50. #define SSB_PCMCIA_CIS_PA_ITSSI 6
  51. #define SSB_PCMCIA_CIS_PA_MAXPOW 7
  52. #define SSB_PCMCIA_CIS_OEMNAME 0x04
  53. #define SSB_PCMCIA_CIS_CCODE 0x05
  54. #define SSB_PCMCIA_CIS_ANTENNA 0x06
  55. #define SSB_PCMCIA_CIS_ANTGAIN 0x07
  56. #define SSB_PCMCIA_CIS_BFLAGS 0x08
  57. #define SSB_PCMCIA_CIS_LEDS 0x09
  58. /* PCMCIA SPROM size. */
  59. #define SSB_PCMCIA_SPROM_SIZE 256
  60. #define SSB_PCMCIA_SPROM_SIZE_BYTES (SSB_PCMCIA_SPROM_SIZE * sizeof(u16))
  61. /* Write to a PCMCIA configuration register. */
  62. static int ssb_pcmcia_cfg_write(struct ssb_bus *bus, u8 offset, u8 value)
  63. {
  64. conf_reg_t reg;
  65. int res;
  66. memset(&reg, 0, sizeof(reg));
  67. reg.Offset = offset;
  68. reg.Action = CS_WRITE;
  69. reg.Value = value;
  70. res = pcmcia_access_configuration_register(bus->host_pcmcia, &reg);
  71. if (unlikely(res != CS_SUCCESS))
  72. return -EBUSY;
  73. return 0;
  74. }
  75. /* Read from a PCMCIA configuration register. */
  76. static int ssb_pcmcia_cfg_read(struct ssb_bus *bus, u8 offset, u8 *value)
  77. {
  78. conf_reg_t reg;
  79. int res;
  80. memset(&reg, 0, sizeof(reg));
  81. reg.Offset = offset;
  82. reg.Action = CS_READ;
  83. res = pcmcia_access_configuration_register(bus->host_pcmcia, &reg);
  84. if (unlikely(res != CS_SUCCESS))
  85. return -EBUSY;
  86. *value = reg.Value;
  87. return 0;
  88. }
  89. int ssb_pcmcia_switch_coreidx(struct ssb_bus *bus,
  90. u8 coreidx)
  91. {
  92. int err;
  93. int attempts = 0;
  94. u32 cur_core;
  95. u32 addr;
  96. u32 read_addr;
  97. u8 val;
  98. addr = (coreidx * SSB_CORE_SIZE) + SSB_ENUM_BASE;
  99. while (1) {
  100. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_ADDRESS0,
  101. (addr & 0x0000F000) >> 12);
  102. if (err)
  103. goto error;
  104. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_ADDRESS1,
  105. (addr & 0x00FF0000) >> 16);
  106. if (err)
  107. goto error;
  108. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_ADDRESS2,
  109. (addr & 0xFF000000) >> 24);
  110. if (err)
  111. goto error;
  112. read_addr = 0;
  113. err = ssb_pcmcia_cfg_read(bus, SSB_PCMCIA_ADDRESS0, &val);
  114. if (err)
  115. goto error;
  116. read_addr |= ((u32)(val & 0x0F)) << 12;
  117. err = ssb_pcmcia_cfg_read(bus, SSB_PCMCIA_ADDRESS1, &val);
  118. if (err)
  119. goto error;
  120. read_addr |= ((u32)val) << 16;
  121. err = ssb_pcmcia_cfg_read(bus, SSB_PCMCIA_ADDRESS2, &val);
  122. if (err)
  123. goto error;
  124. read_addr |= ((u32)val) << 24;
  125. cur_core = (read_addr - SSB_ENUM_BASE) / SSB_CORE_SIZE;
  126. if (cur_core == coreidx)
  127. break;
  128. err = -ETIMEDOUT;
  129. if (attempts++ > SSB_BAR0_MAX_RETRIES)
  130. goto error;
  131. udelay(10);
  132. }
  133. return 0;
  134. error:
  135. ssb_printk(KERN_ERR PFX "Failed to switch to core %u\n", coreidx);
  136. return err;
  137. }
  138. int ssb_pcmcia_switch_core(struct ssb_bus *bus,
  139. struct ssb_device *dev)
  140. {
  141. int err;
  142. #if SSB_VERBOSE_PCMCIACORESWITCH_DEBUG
  143. ssb_printk(KERN_INFO PFX
  144. "Switching to %s core, index %d\n",
  145. ssb_core_name(dev->id.coreid),
  146. dev->core_index);
  147. #endif
  148. err = ssb_pcmcia_switch_coreidx(bus, dev->core_index);
  149. if (!err)
  150. bus->mapped_device = dev;
  151. return err;
  152. }
  153. int ssb_pcmcia_switch_segment(struct ssb_bus *bus, u8 seg)
  154. {
  155. int attempts = 0;
  156. int err;
  157. u8 val;
  158. SSB_WARN_ON((seg != 0) && (seg != 1));
  159. while (1) {
  160. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_MEMSEG, seg);
  161. if (err)
  162. goto error;
  163. err = ssb_pcmcia_cfg_read(bus, SSB_PCMCIA_MEMSEG, &val);
  164. if (err)
  165. goto error;
  166. if (val == seg)
  167. break;
  168. err = -ETIMEDOUT;
  169. if (unlikely(attempts++ > SSB_BAR0_MAX_RETRIES))
  170. goto error;
  171. udelay(10);
  172. }
  173. bus->mapped_pcmcia_seg = seg;
  174. return 0;
  175. error:
  176. ssb_printk(KERN_ERR PFX "Failed to switch pcmcia segment\n");
  177. return err;
  178. }
  179. static int select_core_and_segment(struct ssb_device *dev,
  180. u16 *offset)
  181. {
  182. struct ssb_bus *bus = dev->bus;
  183. int err;
  184. u8 need_segment;
  185. if (*offset >= 0x800) {
  186. *offset -= 0x800;
  187. need_segment = 1;
  188. } else
  189. need_segment = 0;
  190. if (unlikely(dev != bus->mapped_device)) {
  191. err = ssb_pcmcia_switch_core(bus, dev);
  192. if (unlikely(err))
  193. return err;
  194. }
  195. if (unlikely(need_segment != bus->mapped_pcmcia_seg)) {
  196. err = ssb_pcmcia_switch_segment(bus, need_segment);
  197. if (unlikely(err))
  198. return err;
  199. }
  200. return 0;
  201. }
  202. static u8 ssb_pcmcia_read8(struct ssb_device *dev, u16 offset)
  203. {
  204. struct ssb_bus *bus = dev->bus;
  205. unsigned long flags;
  206. int err;
  207. u8 value = 0xFF;
  208. spin_lock_irqsave(&bus->bar_lock, flags);
  209. err = select_core_and_segment(dev, &offset);
  210. if (likely(!err))
  211. value = readb(bus->mmio + offset);
  212. spin_unlock_irqrestore(&bus->bar_lock, flags);
  213. return value;
  214. }
  215. static u16 ssb_pcmcia_read16(struct ssb_device *dev, u16 offset)
  216. {
  217. struct ssb_bus *bus = dev->bus;
  218. unsigned long flags;
  219. int err;
  220. u16 value = 0xFFFF;
  221. spin_lock_irqsave(&bus->bar_lock, flags);
  222. err = select_core_and_segment(dev, &offset);
  223. if (likely(!err))
  224. value = readw(bus->mmio + offset);
  225. spin_unlock_irqrestore(&bus->bar_lock, flags);
  226. return value;
  227. }
  228. static u32 ssb_pcmcia_read32(struct ssb_device *dev, u16 offset)
  229. {
  230. struct ssb_bus *bus = dev->bus;
  231. unsigned long flags;
  232. int err;
  233. u32 lo = 0xFFFFFFFF, hi = 0xFFFFFFFF;
  234. spin_lock_irqsave(&bus->bar_lock, flags);
  235. err = select_core_and_segment(dev, &offset);
  236. if (likely(!err)) {
  237. lo = readw(bus->mmio + offset);
  238. hi = readw(bus->mmio + offset + 2);
  239. }
  240. spin_unlock_irqrestore(&bus->bar_lock, flags);
  241. return (lo | (hi << 16));
  242. }
  243. static void ssb_pcmcia_write8(struct ssb_device *dev, u16 offset, u8 value)
  244. {
  245. struct ssb_bus *bus = dev->bus;
  246. unsigned long flags;
  247. int err;
  248. spin_lock_irqsave(&bus->bar_lock, flags);
  249. err = select_core_and_segment(dev, &offset);
  250. if (likely(!err))
  251. writeb(value, bus->mmio + offset);
  252. mmiowb();
  253. spin_unlock_irqrestore(&bus->bar_lock, flags);
  254. }
  255. static void ssb_pcmcia_write16(struct ssb_device *dev, u16 offset, u16 value)
  256. {
  257. struct ssb_bus *bus = dev->bus;
  258. unsigned long flags;
  259. int err;
  260. spin_lock_irqsave(&bus->bar_lock, flags);
  261. err = select_core_and_segment(dev, &offset);
  262. if (likely(!err))
  263. writew(value, bus->mmio + offset);
  264. mmiowb();
  265. spin_unlock_irqrestore(&bus->bar_lock, flags);
  266. }
  267. static void ssb_pcmcia_write32(struct ssb_device *dev, u16 offset, u32 value)
  268. {
  269. struct ssb_bus *bus = dev->bus;
  270. unsigned long flags;
  271. int err;
  272. spin_lock_irqsave(&bus->bar_lock, flags);
  273. err = select_core_and_segment(dev, &offset);
  274. if (likely(!err)) {
  275. writew((value & 0x0000FFFF), bus->mmio + offset);
  276. writew(((value & 0xFFFF0000) >> 16), bus->mmio + offset + 2);
  277. }
  278. mmiowb();
  279. spin_unlock_irqrestore(&bus->bar_lock, flags);
  280. }
  281. /* Not "static", as it's used in main.c */
  282. const struct ssb_bus_ops ssb_pcmcia_ops = {
  283. .read8 = ssb_pcmcia_read8,
  284. .read16 = ssb_pcmcia_read16,
  285. .read32 = ssb_pcmcia_read32,
  286. .write8 = ssb_pcmcia_write8,
  287. .write16 = ssb_pcmcia_write16,
  288. .write32 = ssb_pcmcia_write32,
  289. };
  290. static int ssb_pcmcia_sprom_command(struct ssb_bus *bus, u8 command)
  291. {
  292. unsigned int i;
  293. int err;
  294. u8 value;
  295. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_SPROMCTL, command);
  296. if (err)
  297. return err;
  298. for (i = 0; i < 1000; i++) {
  299. err = ssb_pcmcia_cfg_read(bus, SSB_PCMCIA_SPROMCTL, &value);
  300. if (err)
  301. return err;
  302. if (value & SSB_PCMCIA_SPROMCTL_DONE)
  303. return 0;
  304. udelay(10);
  305. }
  306. return -ETIMEDOUT;
  307. }
  308. /* offset is the 16bit word offset */
  309. static int ssb_pcmcia_sprom_read(struct ssb_bus *bus, u16 offset, u16 *value)
  310. {
  311. int err;
  312. u8 lo, hi;
  313. offset *= 2; /* Make byte offset */
  314. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_SPROM_ADDRLO,
  315. (offset & 0x00FF));
  316. if (err)
  317. return err;
  318. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_SPROM_ADDRHI,
  319. (offset & 0xFF00) >> 8);
  320. if (err)
  321. return err;
  322. err = ssb_pcmcia_sprom_command(bus, SSB_PCMCIA_SPROMCTL_READ);
  323. if (err)
  324. return err;
  325. err = ssb_pcmcia_cfg_read(bus, SSB_PCMCIA_SPROM_DATALO, &lo);
  326. if (err)
  327. return err;
  328. err = ssb_pcmcia_cfg_read(bus, SSB_PCMCIA_SPROM_DATAHI, &hi);
  329. if (err)
  330. return err;
  331. *value = (lo | (((u16)hi) << 8));
  332. return 0;
  333. }
  334. /* offset is the 16bit word offset */
  335. static int ssb_pcmcia_sprom_write(struct ssb_bus *bus, u16 offset, u16 value)
  336. {
  337. int err;
  338. offset *= 2; /* Make byte offset */
  339. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_SPROM_ADDRLO,
  340. (offset & 0x00FF));
  341. if (err)
  342. return err;
  343. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_SPROM_ADDRHI,
  344. (offset & 0xFF00) >> 8);
  345. if (err)
  346. return err;
  347. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_SPROM_DATALO,
  348. (value & 0x00FF));
  349. if (err)
  350. return err;
  351. err = ssb_pcmcia_cfg_write(bus, SSB_PCMCIA_SPROM_DATAHI,
  352. (value & 0xFF00) >> 8);
  353. if (err)
  354. return err;
  355. err = ssb_pcmcia_sprom_command(bus, SSB_PCMCIA_SPROMCTL_WRITE);
  356. if (err)
  357. return err;
  358. msleep(20);
  359. return 0;
  360. }
  361. /* Read the SPROM image. bufsize is in 16bit words. */
  362. static int ssb_pcmcia_sprom_read_all(struct ssb_bus *bus, u16 *sprom)
  363. {
  364. int err, i;
  365. for (i = 0; i < SSB_PCMCIA_SPROM_SIZE; i++) {
  366. err = ssb_pcmcia_sprom_read(bus, i, &sprom[i]);
  367. if (err)
  368. return err;
  369. }
  370. return 0;
  371. }
  372. /* Write the SPROM image. size is in 16bit words. */
  373. static int ssb_pcmcia_sprom_write_all(struct ssb_bus *bus, const u16 *sprom)
  374. {
  375. int i, err;
  376. bool failed = 0;
  377. size_t size = SSB_PCMCIA_SPROM_SIZE;
  378. ssb_printk(KERN_NOTICE PFX
  379. "Writing SPROM. Do NOT turn off the power! "
  380. "Please stand by...\n");
  381. err = ssb_pcmcia_sprom_command(bus, SSB_PCMCIA_SPROMCTL_WRITEEN);
  382. if (err) {
  383. ssb_printk(KERN_NOTICE PFX
  384. "Could not enable SPROM write access.\n");
  385. return -EBUSY;
  386. }
  387. ssb_printk(KERN_NOTICE PFX "[ 0%%");
  388. msleep(500);
  389. for (i = 0; i < size; i++) {
  390. if (i == size / 4)
  391. ssb_printk("25%%");
  392. else if (i == size / 2)
  393. ssb_printk("50%%");
  394. else if (i == (size * 3) / 4)
  395. ssb_printk("75%%");
  396. else if (i % 2)
  397. ssb_printk(".");
  398. err = ssb_pcmcia_sprom_write(bus, i, sprom[i]);
  399. if (err) {
  400. ssb_printk("\n" KERN_NOTICE PFX
  401. "Failed to write to SPROM.\n");
  402. failed = 1;
  403. break;
  404. }
  405. }
  406. err = ssb_pcmcia_sprom_command(bus, SSB_PCMCIA_SPROMCTL_WRITEDIS);
  407. if (err) {
  408. ssb_printk("\n" KERN_NOTICE PFX
  409. "Could not disable SPROM write access.\n");
  410. failed = 1;
  411. }
  412. msleep(500);
  413. if (!failed) {
  414. ssb_printk("100%% ]\n");
  415. ssb_printk(KERN_NOTICE PFX "SPROM written.\n");
  416. }
  417. return failed ? -EBUSY : 0;
  418. }
  419. static int ssb_pcmcia_sprom_check_crc(const u16 *sprom, size_t size)
  420. {
  421. //TODO
  422. return 0;
  423. }
  424. #define GOTO_ERROR_ON(condition, description) do { \
  425. if (unlikely(condition)) { \
  426. error_description = description; \
  427. goto error; \
  428. } \
  429. } while (0)
  430. int ssb_pcmcia_get_invariants(struct ssb_bus *bus,
  431. struct ssb_init_invariants *iv)
  432. {
  433. tuple_t tuple;
  434. int res;
  435. unsigned char buf[32];
  436. struct ssb_sprom *sprom = &iv->sprom;
  437. struct ssb_boardinfo *bi = &iv->boardinfo;
  438. const char *error_description;
  439. memset(sprom, 0xFF, sizeof(*sprom));
  440. sprom->revision = 1;
  441. sprom->boardflags_lo = 0;
  442. sprom->boardflags_hi = 0;
  443. /* First fetch the MAC address. */
  444. memset(&tuple, 0, sizeof(tuple));
  445. tuple.DesiredTuple = CISTPL_FUNCE;
  446. tuple.TupleData = buf;
  447. tuple.TupleDataMax = sizeof(buf);
  448. res = pcmcia_get_first_tuple(bus->host_pcmcia, &tuple);
  449. GOTO_ERROR_ON(res != CS_SUCCESS, "MAC first tpl");
  450. res = pcmcia_get_tuple_data(bus->host_pcmcia, &tuple);
  451. GOTO_ERROR_ON(res != CS_SUCCESS, "MAC first tpl data");
  452. while (1) {
  453. GOTO_ERROR_ON(tuple.TupleDataLen < 1, "MAC tpl < 1");
  454. if (tuple.TupleData[0] == CISTPL_FUNCE_LAN_NODE_ID)
  455. break;
  456. res = pcmcia_get_next_tuple(bus->host_pcmcia, &tuple);
  457. GOTO_ERROR_ON(res != CS_SUCCESS, "MAC next tpl");
  458. res = pcmcia_get_tuple_data(bus->host_pcmcia, &tuple);
  459. GOTO_ERROR_ON(res != CS_SUCCESS, "MAC next tpl data");
  460. }
  461. GOTO_ERROR_ON(tuple.TupleDataLen != ETH_ALEN + 2, "MAC tpl size");
  462. memcpy(sprom->il0mac, &tuple.TupleData[2], ETH_ALEN);
  463. /* Fetch the vendor specific tuples. */
  464. memset(&tuple, 0, sizeof(tuple));
  465. tuple.DesiredTuple = SSB_PCMCIA_CIS;
  466. tuple.TupleData = buf;
  467. tuple.TupleDataMax = sizeof(buf);
  468. res = pcmcia_get_first_tuple(bus->host_pcmcia, &tuple);
  469. GOTO_ERROR_ON(res != CS_SUCCESS, "VEN first tpl");
  470. res = pcmcia_get_tuple_data(bus->host_pcmcia, &tuple);
  471. GOTO_ERROR_ON(res != CS_SUCCESS, "VEN first tpl data");
  472. while (1) {
  473. GOTO_ERROR_ON(tuple.TupleDataLen < 1, "VEN tpl < 1");
  474. switch (tuple.TupleData[0]) {
  475. case SSB_PCMCIA_CIS_ID:
  476. GOTO_ERROR_ON((tuple.TupleDataLen != 5) &&
  477. (tuple.TupleDataLen != 7),
  478. "id tpl size");
  479. bi->vendor = tuple.TupleData[1] |
  480. ((u16)tuple.TupleData[2] << 8);
  481. break;
  482. case SSB_PCMCIA_CIS_BOARDREV:
  483. GOTO_ERROR_ON(tuple.TupleDataLen != 2,
  484. "boardrev tpl size");
  485. sprom->board_rev = tuple.TupleData[1];
  486. break;
  487. case SSB_PCMCIA_CIS_PA:
  488. GOTO_ERROR_ON(tuple.TupleDataLen != 9,
  489. "pa tpl size");
  490. sprom->pa0b0 = tuple.TupleData[1] |
  491. ((u16)tuple.TupleData[2] << 8);
  492. sprom->pa0b1 = tuple.TupleData[3] |
  493. ((u16)tuple.TupleData[4] << 8);
  494. sprom->pa0b2 = tuple.TupleData[5] |
  495. ((u16)tuple.TupleData[6] << 8);
  496. sprom->itssi_a = tuple.TupleData[7];
  497. sprom->itssi_bg = tuple.TupleData[7];
  498. sprom->maxpwr_a = tuple.TupleData[8];
  499. sprom->maxpwr_bg = tuple.TupleData[8];
  500. break;
  501. case SSB_PCMCIA_CIS_OEMNAME:
  502. /* We ignore this. */
  503. break;
  504. case SSB_PCMCIA_CIS_CCODE:
  505. GOTO_ERROR_ON(tuple.TupleDataLen != 2,
  506. "ccode tpl size");
  507. sprom->country_code = tuple.TupleData[1];
  508. break;
  509. case SSB_PCMCIA_CIS_ANTENNA:
  510. GOTO_ERROR_ON(tuple.TupleDataLen != 2,
  511. "ant tpl size");
  512. sprom->ant_available_a = tuple.TupleData[1];
  513. sprom->ant_available_bg = tuple.TupleData[1];
  514. break;
  515. case SSB_PCMCIA_CIS_ANTGAIN:
  516. GOTO_ERROR_ON(tuple.TupleDataLen != 2,
  517. "antg tpl size");
  518. sprom->antenna_gain.ghz24.a0 = tuple.TupleData[1];
  519. sprom->antenna_gain.ghz24.a1 = tuple.TupleData[1];
  520. sprom->antenna_gain.ghz24.a2 = tuple.TupleData[1];
  521. sprom->antenna_gain.ghz24.a3 = tuple.TupleData[1];
  522. sprom->antenna_gain.ghz5.a0 = tuple.TupleData[1];
  523. sprom->antenna_gain.ghz5.a1 = tuple.TupleData[1];
  524. sprom->antenna_gain.ghz5.a2 = tuple.TupleData[1];
  525. sprom->antenna_gain.ghz5.a3 = tuple.TupleData[1];
  526. break;
  527. case SSB_PCMCIA_CIS_BFLAGS:
  528. GOTO_ERROR_ON(tuple.TupleDataLen != 3,
  529. "bfl tpl size");
  530. sprom->boardflags_lo = tuple.TupleData[1] |
  531. ((u16)tuple.TupleData[2] << 8);
  532. break;
  533. case SSB_PCMCIA_CIS_LEDS:
  534. GOTO_ERROR_ON(tuple.TupleDataLen != 5,
  535. "leds tpl size");
  536. sprom->gpio0 = tuple.TupleData[1];
  537. sprom->gpio1 = tuple.TupleData[2];
  538. sprom->gpio2 = tuple.TupleData[3];
  539. sprom->gpio3 = tuple.TupleData[4];
  540. break;
  541. }
  542. res = pcmcia_get_next_tuple(bus->host_pcmcia, &tuple);
  543. if (res == CS_NO_MORE_ITEMS)
  544. break;
  545. GOTO_ERROR_ON(res != CS_SUCCESS, "VEN next tpl");
  546. res = pcmcia_get_tuple_data(bus->host_pcmcia, &tuple);
  547. GOTO_ERROR_ON(res != CS_SUCCESS, "VEN next tpl data");
  548. }
  549. return 0;
  550. error:
  551. ssb_printk(KERN_ERR PFX
  552. "PCMCIA: Failed to fetch device invariants: %s\n",
  553. error_description);
  554. return -ENODEV;
  555. }
  556. static ssize_t ssb_pcmcia_attr_sprom_show(struct device *pcmciadev,
  557. struct device_attribute *attr,
  558. char *buf)
  559. {
  560. struct pcmcia_device *pdev =
  561. container_of(pcmciadev, struct pcmcia_device, dev);
  562. struct ssb_bus *bus;
  563. bus = ssb_pcmcia_dev_to_bus(pdev);
  564. if (!bus)
  565. return -ENODEV;
  566. return ssb_attr_sprom_show(bus, buf,
  567. ssb_pcmcia_sprom_read_all);
  568. }
  569. static ssize_t ssb_pcmcia_attr_sprom_store(struct device *pcmciadev,
  570. struct device_attribute *attr,
  571. const char *buf, size_t count)
  572. {
  573. struct pcmcia_device *pdev =
  574. container_of(pcmciadev, struct pcmcia_device, dev);
  575. struct ssb_bus *bus;
  576. bus = ssb_pcmcia_dev_to_bus(pdev);
  577. if (!bus)
  578. return -ENODEV;
  579. return ssb_attr_sprom_store(bus, buf, count,
  580. ssb_pcmcia_sprom_check_crc,
  581. ssb_pcmcia_sprom_write_all);
  582. }
  583. static DEVICE_ATTR(ssb_sprom, 0600,
  584. ssb_pcmcia_attr_sprom_show,
  585. ssb_pcmcia_attr_sprom_store);
  586. static int ssb_pcmcia_cor_setup(struct ssb_bus *bus, u8 cor)
  587. {
  588. u8 val;
  589. int err;
  590. err = ssb_pcmcia_cfg_read(bus, cor, &val);
  591. if (err)
  592. return err;
  593. val &= ~COR_SOFT_RESET;
  594. val |= COR_FUNC_ENA | COR_IREQ_ENA | COR_LEVEL_REQ;
  595. err = ssb_pcmcia_cfg_write(bus, cor, val);
  596. if (err)
  597. return err;
  598. msleep(40);
  599. return 0;
  600. }
  601. void ssb_pcmcia_exit(struct ssb_bus *bus)
  602. {
  603. if (bus->bustype != SSB_BUSTYPE_PCMCIA)
  604. return;
  605. device_remove_file(&bus->host_pcmcia->dev, &dev_attr_ssb_sprom);
  606. }
  607. int ssb_pcmcia_init(struct ssb_bus *bus)
  608. {
  609. int err;
  610. if (bus->bustype != SSB_BUSTYPE_PCMCIA)
  611. return 0;
  612. /* Switch segment to a known state and sync
  613. * bus->mapped_pcmcia_seg with hardware state. */
  614. ssb_pcmcia_switch_segment(bus, 0);
  615. /* Init the COR register. */
  616. err = ssb_pcmcia_cor_setup(bus, CISREG_COR);
  617. if (err)
  618. goto error;
  619. /* Some cards also need this register to get poked. */
  620. err = ssb_pcmcia_cor_setup(bus, CISREG_COR + 0x80);
  621. if (err)
  622. goto error;
  623. bus->sprom_size = SSB_PCMCIA_SPROM_SIZE;
  624. mutex_init(&bus->sprom_mutex);
  625. err = device_create_file(&bus->host_pcmcia->dev, &dev_attr_ssb_sprom);
  626. if (err)
  627. goto error;
  628. return 0;
  629. error:
  630. ssb_printk(KERN_ERR PFX "Failed to initialize PCMCIA host device\n");
  631. return err;
  632. }