cistpl.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:40k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /*======================================================================
  2.     PCMCIA Card Information Structure parser
  3.     cistpl.c 1.97 2001/10/04 03:33:49
  4.     The contents of this file are subject to the Mozilla Public
  5.     License Version 1.1 (the "License"); you may not use this file
  6.     except in compliance with the License. You may obtain a copy of
  7.     the License at http://www.mozilla.org/MPL/
  8.     Software distributed under the License is distributed on an "AS
  9.     IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
  10.     implied. See the License for the specific language governing
  11.     rights and limitations under the License.
  12.     The initial developer of the original code is David A. Hinds
  13.     <dahinds@users.sourceforge.net>.  Portions created by David A. Hinds
  14.     are Copyright (C) 1999 David A. Hinds.  All Rights Reserved.
  15.     Alternatively, the contents of this file may be used under the
  16.     terms of the GNU General Public License version 2 (the "GPL"), in
  17.     which case the provisions of the GPL are applicable instead of the
  18.     above.  If you wish to allow the use of your version of this file
  19.     only under the terms of the GPL and not to allow others to use
  20.     your version of this file under the MPL, indicate your decision
  21.     by deleting the provisions above and replace them with the notice
  22.     and other provisions required by the GPL.  If you do not delete
  23.     the provisions above, a recipient may use your version of this
  24.     file under either the MPL or the GPL.
  25.     
  26. ======================================================================*/
  27. #define __NO_VERSION__
  28. #include <linux/config.h>
  29. #include <linux/module.h>
  30. #include <linux/kernel.h>
  31. #include <linux/string.h>
  32. #include <linux/major.h>
  33. #include <linux/errno.h>
  34. #include <linux/timer.h>
  35. #include <linux/slab.h>
  36. #include <linux/mm.h>
  37. #include <linux/sched.h>
  38. #include <linux/pci.h>
  39. #include <linux/ioport.h>
  40. #include <asm/io.h>
  41. #include <asm/byteorder.h>
  42. #include <pcmcia/cs_types.h>
  43. #include <pcmcia/bus_ops.h>
  44. #include <pcmcia/ss.h>
  45. #include <pcmcia/cs.h>
  46. #include <pcmcia/bulkmem.h>
  47. #include <pcmcia/cisreg.h>
  48. #include <pcmcia/cistpl.h>
  49. #include "cs_internal.h"
  50. static const u_char mantissa[] = {
  51.     10, 12, 13, 15, 20, 25, 30, 35,
  52.     40, 45, 50, 55, 60, 70, 80, 90
  53. };
  54. static const u_int exponent[] = {
  55.     1, 10, 100, 1000, 10000, 100000, 1000000, 10000000
  56. };
  57. /* Convert an extended speed byte to a time in nanoseconds */
  58. #define SPEED_CVT(v) 
  59.     (mantissa[(((v)>>3)&15)-1] * exponent[(v)&7] / 10)
  60. /* Convert a power byte to a current in 0.1 microamps */
  61. #define POWER_CVT(v) 
  62.     (mantissa[((v)>>3)&15] * exponent[(v)&7] / 10)
  63. #define POWER_SCALE(v) (exponent[(v)&7])
  64. /* Upper limit on reasonable # of tuples */
  65. #define MAX_TUPLES 200
  66. /*====================================================================*/
  67. /* Parameters that can be set with 'insmod' */
  68. #define INT_MODULE_PARM(n, v) static int n = v; MODULE_PARM(n, "i")
  69. INT_MODULE_PARM(cis_width, 0); /* 16-bit CIS? */
  70. /*======================================================================
  71.     Low-level functions to read and write CIS memory.  I think the
  72.     write routine is only useful for writing one-byte registers.
  73.     
  74. ======================================================================*/
  75. /* Bits in attr field */
  76. #define IS_ATTR 1
  77. #define IS_INDIRECT 8
  78. static int setup_cis_mem(socket_info_t *s);
  79. static void set_cis_map(socket_info_t *s, pccard_mem_map *mem)
  80. {
  81.     s->ss_entry->set_mem_map(s->sock, mem);
  82.     if (s->cap.features & SS_CAP_STATIC_MAP) {
  83. if (s->cis_virt)
  84.     bus_iounmap(s->cap.bus, s->cis_virt);
  85. s->cis_virt = bus_ioremap(s->cap.bus, mem->sys_start,
  86.   s->cap.map_size);
  87.     }
  88. }
  89. void read_cis_mem(socket_info_t *s, int attr, u_int addr,
  90.   u_int len, void *ptr)
  91. {
  92.     pccard_mem_map *mem = &s->cis_mem;
  93.     u_char *sys, *buf = ptr;
  94.     
  95.     DEBUG(3, "cs: read_cis_mem(%d, %#x, %u)n", attr, addr, len);
  96.     if (setup_cis_mem(s) != 0) {
  97. memset(ptr, 0xff, len);
  98. return;
  99.     }
  100.     mem->flags = MAP_ACTIVE | ((cis_width) ? MAP_16BIT : 0);
  101.     if (attr & IS_INDIRECT) {
  102. /* Indirect accesses use a bunch of special registers at fixed
  103.    locations in common memory */
  104. u_char flags = ICTRL0_COMMON|ICTRL0_AUTOINC|ICTRL0_BYTEGRAN;
  105. if (attr & IS_ATTR) { addr *= 2; flags = ICTRL0_AUTOINC; }
  106. mem->card_start = 0; mem->flags = MAP_ACTIVE;
  107. set_cis_map(s, mem);
  108. sys = s->cis_virt;
  109. bus_writeb(s->cap.bus, flags, sys+CISREG_ICTRL0);
  110. bus_writeb(s->cap.bus, addr & 0xff, sys+CISREG_IADDR0);
  111. bus_writeb(s->cap.bus, (addr>>8) & 0xff, sys+CISREG_IADDR1);
  112. bus_writeb(s->cap.bus, (addr>>16) & 0xff, sys+CISREG_IADDR2);
  113. bus_writeb(s->cap.bus, (addr>>24) & 0xff, sys+CISREG_IADDR3);
  114. for ( ; len > 0; len--, buf++)
  115.     *buf = bus_readb(s->cap.bus, sys+CISREG_IDATA0);
  116.     } else {
  117. u_int inc = 1;
  118. if (attr) { mem->flags |= MAP_ATTRIB; inc++; addr *= 2; }
  119. sys += (addr & (s->cap.map_size-1));
  120. mem->card_start = addr & ~(s->cap.map_size-1);
  121. while (len) {
  122.     set_cis_map(s, mem);
  123.     sys = s->cis_virt + (addr & (s->cap.map_size-1));
  124.     for ( ; len > 0; len--, buf++, sys += inc) {
  125. if (sys == s->cis_virt+s->cap.map_size) break;
  126. *buf = bus_readb(s->cap.bus, sys);
  127.     }
  128.     mem->card_start += s->cap.map_size;
  129.     addr = 0;
  130. }
  131.     }
  132.     DEBUG(3, "cs:  %#2.2x %#2.2x %#2.2x %#2.2x ...n",
  133.   *(u_char *)(ptr+0), *(u_char *)(ptr+1),
  134.   *(u_char *)(ptr+2), *(u_char *)(ptr+3));
  135. }
  136. void write_cis_mem(socket_info_t *s, int attr, u_int addr,
  137.    u_int len, void *ptr)
  138. {
  139.     pccard_mem_map *mem = &s->cis_mem;
  140.     u_char *sys, *buf = ptr;
  141.     
  142.     DEBUG(3, "cs: write_cis_mem(%d, %#x, %u)n", attr, addr, len);
  143.     if (setup_cis_mem(s) != 0) return;
  144.     mem->flags = MAP_ACTIVE | ((cis_width) ? MAP_16BIT : 0);
  145.     if (attr & IS_INDIRECT) {
  146. /* Indirect accesses use a bunch of special registers at fixed
  147.    locations in common memory */
  148. u_char flags = ICTRL0_COMMON|ICTRL0_AUTOINC|ICTRL0_BYTEGRAN;
  149. if (attr & IS_ATTR) { addr *= 2; flags = ICTRL0_AUTOINC; }
  150. mem->card_start = 0; mem->flags = MAP_ACTIVE;
  151. set_cis_map(s, mem);
  152. sys = s->cis_virt;
  153. bus_writeb(s->cap.bus, flags, sys+CISREG_ICTRL0);
  154. bus_writeb(s->cap.bus, addr & 0xff, sys+CISREG_IADDR0);
  155. bus_writeb(s->cap.bus, (addr>>8) & 0xff, sys+CISREG_IADDR1);
  156. bus_writeb(s->cap.bus, (addr>>16) & 0xff, sys+CISREG_IADDR2);
  157. bus_writeb(s->cap.bus, (addr>>24) & 0xff, sys+CISREG_IADDR3);
  158. for ( ; len > 0; len--, buf++)
  159.     bus_writeb(s->cap.bus, *buf, sys+CISREG_IDATA0);
  160.     } else {
  161. int inc = 1;
  162. if (attr & IS_ATTR) { mem->flags |= MAP_ATTRIB; inc++; addr *= 2; }
  163. mem->card_start = addr & ~(s->cap.map_size-1);
  164. while (len) {
  165.     set_cis_map(s, mem);
  166.     sys = s->cis_virt + (addr & (s->cap.map_size-1));
  167.     for ( ; len > 0; len--, buf++, sys += inc) {
  168. if (sys == s->cis_virt+s->cap.map_size) break;
  169. bus_writeb(s->cap.bus, *buf, sys);
  170.     }
  171.     mem->card_start += s->cap.map_size;
  172.     addr = 0;
  173. }
  174.     }
  175. }
  176. /*======================================================================
  177.     This is tricky... when we set up CIS memory, we try to validate
  178.     the memory window space allocations.
  179.     
  180. ======================================================================*/
  181. /* Scratch pointer to the socket we use for validation */
  182. static socket_info_t *vs = NULL;
  183. /* Validation function for cards with a valid CIS */
  184. static int cis_readable(u_long base)
  185. {
  186.     cisinfo_t info1, info2;
  187.     int ret;
  188.     vs->cis_mem.sys_start = base;
  189.     vs->cis_mem.sys_stop = base+vs->cap.map_size-1;
  190.     vs->cis_virt = bus_ioremap(vs->cap.bus, base, vs->cap.map_size);
  191.     ret = pcmcia_validate_cis(vs->clients, &info1);
  192.     /* invalidate mapping and CIS cache */
  193.     bus_iounmap(vs->cap.bus, vs->cis_virt); vs->cis_used = 0;
  194.     if ((ret != 0) || (info1.Chains == 0))
  195. return 0;
  196.     vs->cis_mem.sys_start = base+vs->cap.map_size;
  197.     vs->cis_mem.sys_stop = base+2*vs->cap.map_size-1;
  198.     vs->cis_virt = bus_ioremap(vs->cap.bus, base+vs->cap.map_size,
  199.        vs->cap.map_size);
  200.     ret = pcmcia_validate_cis(vs->clients, &info2);
  201.     bus_iounmap(vs->cap.bus, vs->cis_virt); vs->cis_used = 0;
  202.     return ((ret == 0) && (info1.Chains == info2.Chains));
  203. }
  204. /* Validation function for simple memory cards */
  205. static int checksum(u_long base)
  206. {
  207.     int i, a, b, d;
  208.     vs->cis_mem.sys_start = base;
  209.     vs->cis_mem.sys_stop = base+vs->cap.map_size-1;
  210.     vs->cis_virt = bus_ioremap(vs->cap.bus, base, vs->cap.map_size);
  211.     vs->cis_mem.card_start = 0;
  212.     vs->cis_mem.flags = MAP_ACTIVE;
  213.     vs->ss_entry->set_mem_map(vs->sock, &vs->cis_mem);
  214.     /* Don't bother checking every word... */
  215.     a = 0; b = -1;
  216.     for (i = 0; i < vs->cap.map_size; i += 44) {
  217. d = bus_readl(vs->cap.bus, vs->cis_virt+i);
  218. a += d; b &= d;
  219.     }
  220.     bus_iounmap(vs->cap.bus, vs->cis_virt);
  221.     return (b == -1) ? -1 : (a>>1);
  222. }
  223. static int checksum_match(u_long base)
  224. {
  225.     int a = checksum(base), b = checksum(base+vs->cap.map_size);
  226.     return ((a == b) && (a >= 0));
  227. }
  228. static int setup_cis_mem(socket_info_t *s)
  229. {
  230.     if (!(s->cap.features & SS_CAP_STATIC_MAP) &&
  231. (s->cis_mem.sys_start == 0)) {
  232. int low = !(s->cap.features & SS_CAP_PAGE_REGS);
  233. vs = s;
  234. validate_mem(cis_readable, checksum_match, low, s);
  235. s->cis_mem.sys_start = 0;
  236. vs = NULL;
  237. if (find_mem_region(&s->cis_mem.sys_start, s->cap.map_size,
  238.     s->cap.map_size, low, "card services", s)) {
  239.     printk(KERN_NOTICE "cs: unable to map card memory!n");
  240.     return CS_OUT_OF_RESOURCE;
  241. }
  242. s->cis_mem.sys_stop = s->cis_mem.sys_start+s->cap.map_size-1;
  243. s->cis_virt = bus_ioremap(s->cap.bus, s->cis_mem.sys_start,
  244.   s->cap.map_size);
  245.     }
  246.     return 0;
  247. }
  248. void release_cis_mem(socket_info_t *s)
  249. {
  250.     if (s->cis_mem.sys_start != 0) {
  251. s->cis_mem.flags &= ~MAP_ACTIVE;
  252. s->ss_entry->set_mem_map(s->sock, &s->cis_mem);
  253. if (!(s->cap.features & SS_CAP_STATIC_MAP))
  254.     release_mem_resource(s->cis_mem.sys_start, s->cap.map_size);
  255. bus_iounmap(s->cap.bus, s->cis_virt);
  256. s->cis_mem.sys_start = 0;
  257. s->cis_virt = NULL;
  258.     }
  259. }
  260. /*======================================================================
  261.     This is a wrapper around read_cis_mem, with the same interface,
  262.     but which caches information, for cards whose CIS may not be
  263.     readable all the time.
  264.     
  265. ======================================================================*/
  266. static void read_cis_cache(socket_info_t *s, int attr, u_int addr,
  267.    u_int len, void *ptr)
  268. {
  269.     int i;
  270.     char *caddr;
  271.     if (s->fake_cis) {
  272. if (s->fake_cis_len > addr+len)
  273.     memcpy(ptr, s->fake_cis+addr, len);
  274. else
  275.     memset(ptr, 0xff, len);
  276. return;
  277.     }
  278.     caddr = s->cis_cache;
  279.     for (i = 0; i < s->cis_used; i++) {
  280. if ((s->cis_table[i].addr == addr) &&
  281.     (s->cis_table[i].len == len) &&
  282.     (s->cis_table[i].attr == attr)) break;
  283. caddr += s->cis_table[i].len;
  284.     }
  285.     if (i < s->cis_used) {
  286. memcpy(ptr, caddr, len);
  287. return;
  288.     }
  289. #ifdef CONFIG_CARDBUS
  290.     if (s->state & SOCKET_CARDBUS)
  291. read_cb_mem(s, 0, attr, addr, len, ptr);
  292.     else
  293. #endif
  294.     read_cis_mem(s, attr, addr, len, ptr);
  295.     /* Copy data into the cache, if there is room */
  296.     if ((i < MAX_CIS_TABLE) &&
  297. (caddr+len < s->cis_cache+MAX_CIS_DATA)) {
  298. s->cis_table[i].addr = addr;
  299. s->cis_table[i].len = len;
  300. s->cis_table[i].attr = attr;
  301. s->cis_used++;
  302. memcpy(caddr, ptr, len);
  303.     }     
  304. }
  305. /*======================================================================
  306.     This verifies if the CIS of a card matches what is in the CIS
  307.     cache.
  308.     
  309. ======================================================================*/
  310. int verify_cis_cache(socket_info_t *s)
  311. {
  312.     char buf[256], *caddr;
  313.     int i;
  314.     
  315.     caddr = s->cis_cache;
  316.     for (i = 0; i < s->cis_used; i++) {
  317. #ifdef CONFIG_CARDBUS
  318. if (s->state & SOCKET_CARDBUS)
  319.     read_cb_mem(s, 0, s->cis_table[i].attr, s->cis_table[i].addr,
  320. s->cis_table[i].len, buf);
  321. else
  322. #endif
  323.     read_cis_mem(s, s->cis_table[i].attr, s->cis_table[i].addr,
  324.  s->cis_table[i].len, buf);
  325. if (memcmp(buf, caddr, s->cis_table[i].len) != 0)
  326.     break;
  327. caddr += s->cis_table[i].len;
  328.     }
  329.     return (i < s->cis_used);
  330. }
  331. /*======================================================================
  332.     For really bad cards, we provide a facility for uploading a
  333.     replacement CIS.
  334.     
  335. ======================================================================*/
  336. int pcmcia_replace_cis(client_handle_t handle, cisdump_t *cis)
  337. {
  338.     socket_info_t *s;
  339.     if (CHECK_HANDLE(handle))
  340. return CS_BAD_HANDLE;
  341.     s = SOCKET(handle);
  342.     if (s->fake_cis != NULL) {
  343. kfree(s->fake_cis);
  344. s->fake_cis = NULL;
  345.     }
  346.     if (cis->Length > CISTPL_MAX_CIS_SIZE)
  347. return CS_BAD_SIZE;
  348.     s->fake_cis = kmalloc(cis->Length, GFP_KERNEL);
  349.     if (s->fake_cis == NULL)
  350. return CS_OUT_OF_RESOURCE;
  351.     s->fake_cis_len = cis->Length;
  352.     memcpy(s->fake_cis, cis->Data, cis->Length);
  353.     return CS_SUCCESS;
  354. }
  355. /*======================================================================
  356.     The high-level CIS tuple services
  357.     
  358. ======================================================================*/
  359. typedef struct tuple_flags {
  360.     u_int link_space:4;
  361.     u_int has_link:1;
  362.     u_int mfc_fn:3;
  363.     u_int space:4;
  364. } tuple_flags;
  365. #define LINK_SPACE(f) (((tuple_flags *)(&(f)))->link_space)
  366. #define HAS_LINK(f) (((tuple_flags *)(&(f)))->has_link)
  367. #define MFC_FN(f) (((tuple_flags *)(&(f)))->mfc_fn)
  368. #define SPACE(f) (((tuple_flags *)(&(f)))->space)
  369. int pcmcia_get_next_tuple(client_handle_t handle, tuple_t *tuple);
  370. int pcmcia_get_first_tuple(client_handle_t handle, tuple_t *tuple)
  371. {
  372.     socket_info_t *s;
  373.     if (CHECK_HANDLE(handle))
  374. return CS_BAD_HANDLE;
  375.     s = SOCKET(handle);
  376.     if (!(s->state & SOCKET_PRESENT))
  377. return CS_NO_CARD;
  378.     tuple->TupleLink = tuple->Flags = 0;
  379. #ifdef CONFIG_CARDBUS
  380.     if (s->state & SOCKET_CARDBUS) {
  381. u_int ptr;
  382. pcibios_read_config_dword(s->cap.cb_dev->subordinate->number, 0, 0x28, &ptr);
  383. tuple->CISOffset = ptr & ~7;
  384. SPACE(tuple->Flags) = (ptr & 7);
  385.     } else
  386. #endif
  387.     {
  388. /* Assume presence of a LONGLINK_C to address 0 */
  389. tuple->CISOffset = tuple->LinkOffset = 0;
  390. SPACE(tuple->Flags) = HAS_LINK(tuple->Flags) = 1;
  391.     }
  392.     if (!(s->state & SOCKET_CARDBUS) && (s->functions > 1) &&
  393. !(tuple->Attributes & TUPLE_RETURN_COMMON)) {
  394. cisdata_t req = tuple->DesiredTuple;
  395. tuple->DesiredTuple = CISTPL_LONGLINK_MFC;
  396. if (pcmcia_get_next_tuple(handle, tuple) == CS_SUCCESS) {
  397.     tuple->DesiredTuple = CISTPL_LINKTARGET;
  398.     if (pcmcia_get_next_tuple(handle, tuple) != CS_SUCCESS)
  399. return CS_NO_MORE_ITEMS;
  400. } else
  401.     tuple->CISOffset = tuple->TupleLink = 0;
  402. tuple->DesiredTuple = req;
  403.     }
  404.     return pcmcia_get_next_tuple(handle, tuple);
  405. }
  406. static int follow_link(socket_info_t *s, tuple_t *tuple)
  407. {
  408.     u_char link[5];
  409.     u_int ofs;
  410.     if (MFC_FN(tuple->Flags)) {
  411. /* Get indirect link from the MFC tuple */
  412. read_cis_cache(s, LINK_SPACE(tuple->Flags),
  413.        tuple->LinkOffset, 5, link);
  414. ofs = le32_to_cpu(*(u_int *)(link+1));
  415. SPACE(tuple->Flags) = (link[0] == CISTPL_MFC_ATTR);
  416. /* Move to the next indirect link */
  417. tuple->LinkOffset += 5;
  418. MFC_FN(tuple->Flags)--;
  419.     } else if (HAS_LINK(tuple->Flags)) {
  420. ofs = tuple->LinkOffset;
  421. SPACE(tuple->Flags) = LINK_SPACE(tuple->Flags);
  422. HAS_LINK(tuple->Flags) = 0;
  423.     } else {
  424. return -1;
  425.     }
  426.     if (!(s->state & SOCKET_CARDBUS) && SPACE(tuple->Flags)) {
  427. /* This is ugly, but a common CIS error is to code the long
  428.    link offset incorrectly, so we check the right spot... */
  429. read_cis_cache(s, SPACE(tuple->Flags), ofs, 5, link);
  430. if ((link[0] == CISTPL_LINKTARGET) && (link[1] >= 3) &&
  431.     (strncmp(link+2, "CIS", 3) == 0))
  432.     return ofs;
  433. /* Then, we try the wrong spot... */
  434. ofs = ofs >> 1;
  435.     }
  436.     read_cis_cache(s, SPACE(tuple->Flags), ofs, 5, link);
  437.     if ((link[0] != CISTPL_LINKTARGET) || (link[1] < 3) ||
  438. (strncmp(link+2, "CIS", 3) != 0))
  439. return -1;
  440.     return ofs;
  441. }
  442. int pcmcia_get_next_tuple(client_handle_t handle, tuple_t *tuple)
  443. {
  444.     socket_info_t *s;
  445.     u_char link[2], tmp;
  446.     int ofs, i, attr;
  447.     
  448.     if (CHECK_HANDLE(handle))
  449. return CS_BAD_HANDLE;
  450.     s = SOCKET(handle);
  451.     if (!(s->state & SOCKET_PRESENT))
  452. return CS_NO_CARD;
  453.     link[1] = tuple->TupleLink;
  454.     ofs = tuple->CISOffset + tuple->TupleLink;
  455.     attr = SPACE(tuple->Flags);
  456.     for (i = 0; i < MAX_TUPLES; i++) {
  457. if (link[1] == 0xff) {
  458.     link[0] = CISTPL_END;
  459. } else {
  460.     read_cis_cache(s, attr, ofs, 2, link);
  461.     if (link[0] == CISTPL_NULL) {
  462. ofs++; continue;
  463.     }
  464. }
  465. /* End of chain?  Follow long link if possible */
  466. if (link[0] == CISTPL_END) {
  467.     if ((ofs = follow_link(s, tuple)) < 0)
  468. return CS_NO_MORE_ITEMS;
  469.     attr = SPACE(tuple->Flags);
  470.     read_cis_cache(s, attr, ofs, 2, link);
  471. }
  472. /* Is this a link tuple?  Make a note of it */
  473. if ((link[0] == CISTPL_LONGLINK_A) ||
  474.     (link[0] == CISTPL_LONGLINK_C) ||
  475.     (link[0] == CISTPL_LONGLINK_MFC) ||
  476.     (link[0] == CISTPL_LINKTARGET) ||
  477.     (link[0] == CISTPL_INDIRECT) ||
  478.     (link[0] == CISTPL_NO_LINK)) {
  479.     switch (link[0]) {
  480.     case CISTPL_LONGLINK_A:
  481. HAS_LINK(tuple->Flags) = 1;
  482. LINK_SPACE(tuple->Flags) = attr | IS_ATTR;
  483. read_cis_cache(s, attr, ofs+2, 4, &tuple->LinkOffset);
  484. break;
  485.     case CISTPL_LONGLINK_C:
  486. HAS_LINK(tuple->Flags) = 1;
  487. LINK_SPACE(tuple->Flags) = attr & ~IS_ATTR;
  488. read_cis_cache(s, attr, ofs+2, 4, &tuple->LinkOffset);
  489. break;
  490.     case CISTPL_INDIRECT:
  491. HAS_LINK(tuple->Flags) = 1;
  492. LINK_SPACE(tuple->Flags) = IS_ATTR | IS_INDIRECT;
  493. tuple->LinkOffset = 0;
  494. break;
  495.     case CISTPL_LONGLINK_MFC:
  496. tuple->LinkOffset = ofs + 3;
  497. LINK_SPACE(tuple->Flags) = attr;
  498. if (handle->Function == BIND_FN_ALL) {
  499.     /* Follow all the MFC links */
  500.     read_cis_cache(s, attr, ofs+2, 1, &tmp);
  501.     MFC_FN(tuple->Flags) = tmp;
  502. } else {
  503.     /* Follow exactly one of the links */
  504.     MFC_FN(tuple->Flags) = 1;
  505.     tuple->LinkOffset += handle->Function * 5;
  506. }
  507. break;
  508.     case CISTPL_NO_LINK:
  509. HAS_LINK(tuple->Flags) = 0;
  510. break;
  511.     }
  512.     if ((tuple->Attributes & TUPLE_RETURN_LINK) &&
  513. (tuple->DesiredTuple == RETURN_FIRST_TUPLE))
  514. break;
  515. } else
  516.     if (tuple->DesiredTuple == RETURN_FIRST_TUPLE)
  517. break;
  518. if (link[0] == tuple->DesiredTuple)
  519.     break;
  520. ofs += link[1] + 2;
  521.     }
  522.     if (i == MAX_TUPLES) {
  523. DEBUG(1, "cs: overrun in pcmcia_get_next_tuple for socket %dn",
  524.       handle->Socket);
  525. return CS_NO_MORE_ITEMS;
  526.     }
  527.     
  528.     tuple->TupleCode = link[0];
  529.     tuple->TupleLink = link[1];
  530.     tuple->CISOffset = ofs + 2;
  531.     return CS_SUCCESS;
  532. }
  533. /*====================================================================*/
  534. #define _MIN(a, b) (((a) < (b)) ? (a) : (b))
  535. int pcmcia_get_tuple_data(client_handle_t handle, tuple_t *tuple)
  536. {
  537.     socket_info_t *s;
  538.     u_int len;
  539.     
  540.     if (CHECK_HANDLE(handle))
  541. return CS_BAD_HANDLE;
  542.     s = SOCKET(handle);
  543.     if (tuple->TupleLink < tuple->TupleOffset)
  544. return CS_NO_MORE_ITEMS;
  545.     len = tuple->TupleLink - tuple->TupleOffset;
  546.     tuple->TupleDataLen = tuple->TupleLink;
  547.     if (len == 0)
  548. return CS_SUCCESS;
  549.     read_cis_cache(s, SPACE(tuple->Flags),
  550.    tuple->CISOffset + tuple->TupleOffset,
  551.    _MIN(len, tuple->TupleDataMax), tuple->TupleData);
  552.     return CS_SUCCESS;
  553. }
  554. /*======================================================================
  555.     Parsing routines for individual tuples
  556.     
  557. ======================================================================*/
  558. static int parse_device(tuple_t *tuple, cistpl_device_t *device)
  559. {
  560.     int i;
  561.     u_char scale;
  562.     u_char *p, *q;
  563.     p = (u_char *)tuple->TupleData;
  564.     q = p + tuple->TupleDataLen;
  565.     device->ndev = 0;
  566.     for (i = 0; i < CISTPL_MAX_DEVICES; i++) {
  567. if (*p == 0xff) break;
  568. device->dev[i].type = (*p >> 4);
  569. device->dev[i].wp = (*p & 0x08) ? 1 : 0;
  570. switch (*p & 0x07) {
  571. case 0: device->dev[i].speed = 0;   break;
  572. case 1: device->dev[i].speed = 250; break;
  573. case 2: device->dev[i].speed = 200; break;
  574. case 3: device->dev[i].speed = 150; break;
  575. case 4: device->dev[i].speed = 100; break;
  576. case 7:
  577.     if (++p == q) return CS_BAD_TUPLE;
  578.     device->dev[i].speed = SPEED_CVT(*p);
  579.     while (*p & 0x80)
  580. if (++p == q) return CS_BAD_TUPLE;
  581.     break;
  582. default:
  583.     return CS_BAD_TUPLE;
  584. }
  585. if (++p == q) return CS_BAD_TUPLE;
  586. if (*p == 0xff) break;
  587. scale = *p & 7;
  588. if (scale == 7) return CS_BAD_TUPLE;
  589. device->dev[i].size = ((*p >> 3) + 1) * (512 << (scale*2));
  590. device->ndev++;
  591. if (++p == q) break;
  592.     }
  593.     
  594.     return CS_SUCCESS;
  595. }
  596. /*====================================================================*/
  597. static int parse_checksum(tuple_t *tuple, cistpl_checksum_t *csum)
  598. {
  599.     u_char *p;
  600.     if (tuple->TupleDataLen < 5)
  601. return CS_BAD_TUPLE;
  602.     p = (u_char *)tuple->TupleData;
  603.     csum->addr = tuple->CISOffset+(short)le16_to_cpu(*(u_short *)p)-2;
  604.     csum->len = le16_to_cpu(*(u_short *)(p + 2));
  605.     csum->sum = *(p+4);
  606.     return CS_SUCCESS;
  607. }
  608. /*====================================================================*/
  609. static int parse_longlink(tuple_t *tuple, cistpl_longlink_t *link)
  610. {
  611.     if (tuple->TupleDataLen < 4)
  612. return CS_BAD_TUPLE;
  613.     link->addr = le32_to_cpu(*(u_int *)tuple->TupleData);
  614.     return CS_SUCCESS;
  615. }
  616. /*====================================================================*/
  617. static int parse_longlink_mfc(tuple_t *tuple,
  618.       cistpl_longlink_mfc_t *link)
  619. {
  620.     u_char *p;
  621.     int i;
  622.     
  623.     p = (u_char *)tuple->TupleData;
  624.     
  625.     link->nfn = *p; p++;
  626.     if (tuple->TupleDataLen <= link->nfn*5)
  627. return CS_BAD_TUPLE;
  628.     for (i = 0; i < link->nfn; i++) {
  629. link->fn[i].space = *p; p++;
  630. link->fn[i].addr = le32_to_cpu(*(u_int *)p); p += 4;
  631.     }
  632.     return CS_SUCCESS;
  633. }
  634. /*====================================================================*/
  635. static int parse_strings(u_char *p, u_char *q, int max,
  636.  char *s, u_char *ofs, u_char *found)
  637. {
  638.     int i, j, ns;
  639.     if (p == q) return CS_BAD_TUPLE;
  640.     ns = 0; j = 0;
  641.     for (i = 0; i < max; i++) {
  642. if (*p == 0xff) break;
  643. ofs[i] = j;
  644. ns++;
  645. for (;;) {
  646.     s[j++] = (*p == 0xff) ? '' : *p;
  647.     if ((*p == '') || (*p == 0xff)) break;
  648.     if (++p == q) return CS_BAD_TUPLE;
  649. }
  650. if ((*p == 0xff) || (++p == q)) break;
  651.     }
  652.     if (found) {
  653. *found = ns;
  654. return CS_SUCCESS;
  655.     } else {
  656. return (ns == max) ? CS_SUCCESS : CS_BAD_TUPLE;
  657.     }
  658. }
  659. /*====================================================================*/
  660. static int parse_vers_1(tuple_t *tuple, cistpl_vers_1_t *vers_1)
  661. {
  662.     u_char *p, *q;
  663.     
  664.     p = (u_char *)tuple->TupleData;
  665.     q = p + tuple->TupleDataLen;
  666.     
  667.     vers_1->major = *p; p++;
  668.     vers_1->minor = *p; p++;
  669.     if (p >= q) return CS_BAD_TUPLE;
  670.     return parse_strings(p, q, CISTPL_VERS_1_MAX_PROD_STRINGS,
  671.  vers_1->str, vers_1->ofs, &vers_1->ns);
  672. }
  673. /*====================================================================*/
  674. static int parse_altstr(tuple_t *tuple, cistpl_altstr_t *altstr)
  675. {
  676.     u_char *p, *q;
  677.     
  678.     p = (u_char *)tuple->TupleData;
  679.     q = p + tuple->TupleDataLen;
  680.     
  681.     return parse_strings(p, q, CISTPL_MAX_ALTSTR_STRINGS,
  682.  altstr->str, altstr->ofs, &altstr->ns);
  683. }
  684. /*====================================================================*/
  685. static int parse_jedec(tuple_t *tuple, cistpl_jedec_t *jedec)
  686. {
  687.     u_char *p, *q;
  688.     int nid;
  689.     p = (u_char *)tuple->TupleData;
  690.     q = p + tuple->TupleDataLen;
  691.     for (nid = 0; nid < CISTPL_MAX_DEVICES; nid++) {
  692. if (p > q-2) break;
  693. jedec->id[nid].mfr = p[0];
  694. jedec->id[nid].info = p[1];
  695. p += 2;
  696.     }
  697.     jedec->nid = nid;
  698.     return CS_SUCCESS;
  699. }
  700. /*====================================================================*/
  701. static int parse_manfid(tuple_t *tuple, cistpl_manfid_t *m)
  702. {
  703.     u_short *p;
  704.     if (tuple->TupleDataLen < 4)
  705. return CS_BAD_TUPLE;
  706.     p = (u_short *)tuple->TupleData;
  707.     m->manf = le16_to_cpu(p[0]);
  708.     m->card = le16_to_cpu(p[1]);
  709.     return CS_SUCCESS;
  710. }
  711. /*====================================================================*/
  712. static int parse_funcid(tuple_t *tuple, cistpl_funcid_t *f)
  713. {
  714.     u_char *p;
  715.     if (tuple->TupleDataLen < 2)
  716. return CS_BAD_TUPLE;
  717.     p = (u_char *)tuple->TupleData;
  718.     f->func = p[0];
  719.     f->sysinit = p[1];
  720.     return CS_SUCCESS;
  721. }
  722. /*====================================================================*/
  723. static int parse_funce(tuple_t *tuple, cistpl_funce_t *f)
  724. {
  725.     u_char *p;
  726.     int i;
  727.     if (tuple->TupleDataLen < 1)
  728. return CS_BAD_TUPLE;
  729.     p = (u_char *)tuple->TupleData;
  730.     f->type = p[0];
  731.     for (i = 1; i < tuple->TupleDataLen; i++)
  732. f->data[i-1] = p[i];
  733.     return CS_SUCCESS;
  734. }
  735. /*====================================================================*/
  736. static int parse_config(tuple_t *tuple, cistpl_config_t *config)
  737. {
  738.     int rasz, rmsz, i;
  739.     u_char *p;
  740.     p = (u_char *)tuple->TupleData;
  741.     rasz = *p & 0x03;
  742.     rmsz = (*p & 0x3c) >> 2;
  743.     if (tuple->TupleDataLen < rasz+rmsz+4)
  744. return CS_BAD_TUPLE;
  745.     config->last_idx = *(++p);
  746.     p++;
  747.     config->base = 0;
  748.     for (i = 0; i <= rasz; i++)
  749. config->base += p[i] << (8*i);
  750.     p += rasz+1;
  751.     for (i = 0; i < 4; i++)
  752. config->rmask[i] = 0;
  753.     for (i = 0; i <= rmsz; i++)
  754. config->rmask[i>>2] += p[i] << (8*(i%4));
  755.     config->subtuples = tuple->TupleDataLen - (rasz+rmsz+4);
  756.     return CS_SUCCESS;
  757. }
  758. /*======================================================================
  759.     The following routines are all used to parse the nightmarish
  760.     config table entries.
  761.     
  762. ======================================================================*/
  763. static u_char *parse_power(u_char *p, u_char *q,
  764.    cistpl_power_t *pwr)
  765. {
  766.     int i;
  767.     u_int scale;
  768.     if (p == q) return NULL;
  769.     pwr->present = *p;
  770.     pwr->flags = 0;
  771.     p++;
  772.     for (i = 0; i < 7; i++)
  773. if (pwr->present & (1<<i)) {
  774.     if (p == q) return NULL;
  775.     pwr->param[i] = POWER_CVT(*p);
  776.     scale = POWER_SCALE(*p);
  777.     while (*p & 0x80) {
  778. if (++p == q) return NULL;
  779. if ((*p & 0x7f) < 100)
  780.     pwr->param[i] += (*p & 0x7f) * scale / 100;
  781. else if (*p == 0x7d)
  782.     pwr->flags |= CISTPL_POWER_HIGHZ_OK;
  783. else if (*p == 0x7e)
  784.     pwr->param[i] = 0;
  785. else if (*p == 0x7f)
  786.     pwr->flags |= CISTPL_POWER_HIGHZ_REQ;
  787. else
  788.     return NULL;
  789.     }
  790.     p++;
  791. }
  792.     return p;
  793. }
  794. /*====================================================================*/
  795. static u_char *parse_timing(u_char *p, u_char *q,
  796.     cistpl_timing_t *timing)
  797. {
  798.     u_char scale;
  799.     if (p == q) return NULL;
  800.     scale = *p;
  801.     if ((scale & 3) != 3) {
  802. if (++p == q) return NULL;
  803. timing->wait = SPEED_CVT(*p);
  804. timing->waitscale = exponent[scale & 3];
  805.     } else
  806. timing->wait = 0;
  807.     scale >>= 2;
  808.     if ((scale & 7) != 7) {
  809. if (++p == q) return NULL;
  810. timing->ready = SPEED_CVT(*p);
  811. timing->rdyscale = exponent[scale & 7];
  812.     } else
  813. timing->ready = 0;
  814.     scale >>= 3;
  815.     if (scale != 7) {
  816. if (++p == q) return NULL;
  817. timing->reserved = SPEED_CVT(*p);
  818. timing->rsvscale = exponent[scale];
  819.     } else
  820. timing->reserved = 0;
  821.     p++;
  822.     return p;
  823. }
  824. /*====================================================================*/
  825. static u_char *parse_io(u_char *p, u_char *q, cistpl_io_t *io)
  826. {
  827.     int i, j, bsz, lsz;
  828.     if (p == q) return NULL;
  829.     io->flags = *p;
  830.     if (!(*p & 0x80)) {
  831. io->nwin = 1;
  832. io->win[0].base = 0;
  833. io->win[0].len = (1 << (io->flags & CISTPL_IO_LINES_MASK));
  834. return p+1;
  835.     }
  836.     
  837.     if (++p == q) return NULL;
  838.     io->nwin = (*p & 0x0f) + 1;
  839.     bsz = (*p & 0x30) >> 4;
  840.     if (bsz == 3) bsz++;
  841.     lsz = (*p & 0xc0) >> 6;
  842.     if (lsz == 3) lsz++;
  843.     p++;
  844.     
  845.     for (i = 0; i < io->nwin; i++) {
  846. io->win[i].base = 0;
  847. io->win[i].len = 1;
  848. for (j = 0; j < bsz; j++, p++) {
  849.     if (p == q) return NULL;
  850.     io->win[i].base += *p << (j*8);
  851. }
  852. for (j = 0; j < lsz; j++, p++) {
  853.     if (p == q) return NULL;
  854.     io->win[i].len += *p << (j*8);
  855. }
  856.     }
  857.     return p;
  858. }
  859. /*====================================================================*/
  860. static u_char *parse_mem(u_char *p, u_char *q, cistpl_mem_t *mem)
  861. {
  862.     int i, j, asz, lsz, has_ha;
  863.     u_int len, ca, ha;
  864.     if (p == q) return NULL;
  865.     mem->nwin = (*p & 0x07) + 1;
  866.     lsz = (*p & 0x18) >> 3;
  867.     asz = (*p & 0x60) >> 5;
  868.     has_ha = (*p & 0x80);
  869.     if (++p == q) return NULL;
  870.     
  871.     for (i = 0; i < mem->nwin; i++) {
  872. len = ca = ha = 0;
  873. for (j = 0; j < lsz; j++, p++) {
  874.     if (p == q) return NULL;
  875.     len += *p << (j*8);
  876. }
  877. for (j = 0; j < asz; j++, p++) {
  878.     if (p == q) return NULL;
  879.     ca += *p << (j*8);
  880. }
  881. if (has_ha)
  882.     for (j = 0; j < asz; j++, p++) {
  883. if (p == q) return NULL;
  884. ha += *p << (j*8);
  885.     }
  886. mem->win[i].len = len << 8;
  887. mem->win[i].card_addr = ca << 8;
  888. mem->win[i].host_addr = ha << 8;
  889.     }
  890.     return p;
  891. }
  892. /*====================================================================*/
  893. static u_char *parse_irq(u_char *p, u_char *q, cistpl_irq_t *irq)
  894. {
  895.     if (p == q) return NULL;
  896.     irq->IRQInfo1 = *p; p++;
  897.     if (irq->IRQInfo1 & IRQ_INFO2_VALID) {
  898. if (p+2 > q) return NULL;
  899. irq->IRQInfo2 = (p[1]<<8) + p[0];
  900. p += 2;
  901.     }
  902.     return p;
  903. }
  904. /*====================================================================*/
  905. static int parse_cftable_entry(tuple_t *tuple,
  906.        cistpl_cftable_entry_t *entry)
  907. {
  908.     u_char *p, *q, features;
  909.     p = tuple->TupleData;
  910.     q = p + tuple->TupleDataLen;
  911.     entry->index = *p & 0x3f;
  912.     entry->flags = 0;
  913.     if (*p & 0x40)
  914. entry->flags |= CISTPL_CFTABLE_DEFAULT;
  915.     if (*p & 0x80) {
  916. if (++p == q) return CS_BAD_TUPLE;
  917. if (*p & 0x10)
  918.     entry->flags |= CISTPL_CFTABLE_BVDS;
  919. if (*p & 0x20)
  920.     entry->flags |= CISTPL_CFTABLE_WP;
  921. if (*p & 0x40)
  922.     entry->flags |= CISTPL_CFTABLE_RDYBSY;
  923. if (*p & 0x80)
  924.     entry->flags |= CISTPL_CFTABLE_MWAIT;
  925. entry->interface = *p & 0x0f;
  926.     } else
  927. entry->interface = 0;
  928.     /* Process optional features */
  929.     if (++p == q) return CS_BAD_TUPLE;
  930.     features = *p; p++;
  931.     /* Power options */
  932.     if ((features & 3) > 0) {
  933. p = parse_power(p, q, &entry->vcc);
  934. if (p == NULL) return CS_BAD_TUPLE;
  935.     } else
  936. entry->vcc.present = 0;
  937.     if ((features & 3) > 1) {
  938. p = parse_power(p, q, &entry->vpp1);
  939. if (p == NULL) return CS_BAD_TUPLE;
  940.     } else
  941. entry->vpp1.present = 0;
  942.     if ((features & 3) > 2) {
  943. p = parse_power(p, q, &entry->vpp2);
  944. if (p == NULL) return CS_BAD_TUPLE;
  945.     } else
  946. entry->vpp2.present = 0;
  947.     /* Timing options */
  948.     if (features & 0x04) {
  949. p = parse_timing(p, q, &entry->timing);
  950. if (p == NULL) return CS_BAD_TUPLE;
  951.     } else {
  952. entry->timing.wait = 0;
  953. entry->timing.ready = 0;
  954. entry->timing.reserved = 0;
  955.     }
  956.     
  957.     /* I/O window options */
  958.     if (features & 0x08) {
  959. p = parse_io(p, q, &entry->io);
  960. if (p == NULL) return CS_BAD_TUPLE;
  961.     } else
  962. entry->io.nwin = 0;
  963.     
  964.     /* Interrupt options */
  965.     if (features & 0x10) {
  966. p = parse_irq(p, q, &entry->irq);
  967. if (p == NULL) return CS_BAD_TUPLE;
  968.     } else
  969. entry->irq.IRQInfo1 = 0;
  970.     switch (features & 0x60) {
  971.     case 0x00:
  972. entry->mem.nwin = 0;
  973. break;
  974.     case 0x20:
  975. entry->mem.nwin = 1;
  976. entry->mem.win[0].len = le16_to_cpu(*(u_short *)p) << 8;
  977. entry->mem.win[0].card_addr = 0;
  978. entry->mem.win[0].host_addr = 0;
  979. p += 2;
  980. if (p > q) return CS_BAD_TUPLE;
  981. break;
  982.     case 0x40:
  983. entry->mem.nwin = 1;
  984. entry->mem.win[0].len = le16_to_cpu(*(u_short *)p) << 8;
  985. entry->mem.win[0].card_addr =
  986.     le16_to_cpu(*(u_short *)(p+2)) << 8;
  987. entry->mem.win[0].host_addr = 0;
  988. p += 4;
  989. if (p > q) return CS_BAD_TUPLE;
  990. break;
  991.     case 0x60:
  992. p = parse_mem(p, q, &entry->mem);
  993. if (p == NULL) return CS_BAD_TUPLE;
  994. break;
  995.     }
  996.     /* Misc features */
  997.     if (features & 0x80) {
  998. if (p == q) return CS_BAD_TUPLE;
  999. entry->flags |= (*p << 8);
  1000. while (*p & 0x80)
  1001.     if (++p == q) return CS_BAD_TUPLE;
  1002. p++;
  1003.     }
  1004.     entry->subtuples = q-p;
  1005.     
  1006.     return CS_SUCCESS;
  1007. }
  1008. /*====================================================================*/
  1009. #ifdef CONFIG_CARDBUS
  1010. static int parse_bar(tuple_t *tuple, cistpl_bar_t *bar)
  1011. {
  1012.     u_char *p;
  1013.     if (tuple->TupleDataLen < 6)
  1014. return CS_BAD_TUPLE;
  1015.     p = (u_char *)tuple->TupleData;
  1016.     bar->attr = *p;
  1017.     p += 2;
  1018.     bar->size = le32_to_cpu(*(u_int *)p);
  1019.     return CS_SUCCESS;
  1020. }
  1021. static int parse_config_cb(tuple_t *tuple, cistpl_config_t *config)
  1022. {
  1023.     u_char *p;
  1024.     
  1025.     p = (u_char *)tuple->TupleData;
  1026.     if ((*p != 3) || (tuple->TupleDataLen < 6))
  1027. return CS_BAD_TUPLE;
  1028.     config->last_idx = *(++p);
  1029.     p++;
  1030.     config->base = le32_to_cpu(*(u_int *)p);
  1031.     config->subtuples = tuple->TupleDataLen - 6;
  1032.     return CS_SUCCESS;
  1033. }
  1034. static int parse_cftable_entry_cb(tuple_t *tuple,
  1035.   cistpl_cftable_entry_cb_t *entry)
  1036. {
  1037.     u_char *p, *q, features;
  1038.     p = tuple->TupleData;
  1039.     q = p + tuple->TupleDataLen;
  1040.     entry->index = *p & 0x3f;
  1041.     entry->flags = 0;
  1042.     if (*p & 0x40)
  1043. entry->flags |= CISTPL_CFTABLE_DEFAULT;
  1044.     /* Process optional features */
  1045.     if (++p == q) return CS_BAD_TUPLE;
  1046.     features = *p; p++;
  1047.     /* Power options */
  1048.     if ((features & 3) > 0) {
  1049. p = parse_power(p, q, &entry->vcc);
  1050. if (p == NULL) return CS_BAD_TUPLE;
  1051.     } else
  1052. entry->vcc.present = 0;
  1053.     if ((features & 3) > 1) {
  1054. p = parse_power(p, q, &entry->vpp1);
  1055. if (p == NULL) return CS_BAD_TUPLE;
  1056.     } else
  1057. entry->vpp1.present = 0;
  1058.     if ((features & 3) > 2) {
  1059. p = parse_power(p, q, &entry->vpp2);
  1060. if (p == NULL) return CS_BAD_TUPLE;
  1061.     } else
  1062. entry->vpp2.present = 0;
  1063.     /* I/O window options */
  1064.     if (features & 0x08) {
  1065. if (p == q) return CS_BAD_TUPLE;
  1066. entry->io = *p; p++;
  1067.     } else
  1068. entry->io = 0;
  1069.     
  1070.     /* Interrupt options */
  1071.     if (features & 0x10) {
  1072. p = parse_irq(p, q, &entry->irq);
  1073. if (p == NULL) return CS_BAD_TUPLE;
  1074.     } else
  1075. entry->irq.IRQInfo1 = 0;
  1076.     if (features & 0x20) {
  1077. if (p == q) return CS_BAD_TUPLE;
  1078. entry->mem = *p; p++;
  1079.     } else
  1080. entry->mem = 0;
  1081.     /* Misc features */
  1082.     if (features & 0x80) {
  1083. if (p == q) return CS_BAD_TUPLE;
  1084. entry->flags |= (*p << 8);
  1085. if (*p & 0x80) {
  1086.     if (++p == q) return CS_BAD_TUPLE;
  1087.     entry->flags |= (*p << 16);
  1088. }
  1089. while (*p & 0x80)
  1090.     if (++p == q) return CS_BAD_TUPLE;
  1091. p++;
  1092.     }
  1093.     entry->subtuples = q-p;
  1094.     
  1095.     return CS_SUCCESS;
  1096. }
  1097. #endif
  1098. /*====================================================================*/
  1099. static int parse_device_geo(tuple_t *tuple, cistpl_device_geo_t *geo)
  1100. {
  1101.     u_char *p, *q;
  1102.     int n;
  1103.     p = (u_char *)tuple->TupleData;
  1104.     q = p + tuple->TupleDataLen;
  1105.     for (n = 0; n < CISTPL_MAX_DEVICES; n++) {
  1106. if (p > q-6) break;
  1107. geo->geo[n].buswidth = p[0];
  1108. geo->geo[n].erase_block = 1 << (p[1]-1);
  1109. geo->geo[n].read_block  = 1 << (p[2]-1);
  1110. geo->geo[n].write_block = 1 << (p[3]-1);
  1111. geo->geo[n].partition   = 1 << (p[4]-1);
  1112. geo->geo[n].interleave  = 1 << (p[5]-1);
  1113. p += 6;
  1114.     }
  1115.     geo->ngeo = n;
  1116.     return CS_SUCCESS;
  1117. }
  1118. /*====================================================================*/
  1119. static int parse_vers_2(tuple_t *tuple, cistpl_vers_2_t *v2)
  1120. {
  1121.     u_char *p, *q;
  1122.     if (tuple->TupleDataLen < 10)
  1123. return CS_BAD_TUPLE;
  1124.     
  1125.     p = tuple->TupleData;
  1126.     q = p + tuple->TupleDataLen;
  1127.     v2->vers = p[0];
  1128.     v2->comply = p[1];
  1129.     v2->dindex = le16_to_cpu(*(u_short *)(p+2));
  1130.     v2->vspec8 = p[6];
  1131.     v2->vspec9 = p[7];
  1132.     v2->nhdr = p[8];
  1133.     p += 9;
  1134.     return parse_strings(p, q, 2, v2->str, &v2->vendor, NULL);
  1135. }
  1136. /*====================================================================*/
  1137. static int parse_org(tuple_t *tuple, cistpl_org_t *org)
  1138. {
  1139.     u_char *p, *q;
  1140.     int i;
  1141.     
  1142.     p = tuple->TupleData;
  1143.     q = p + tuple->TupleDataLen;
  1144.     if (p == q) return CS_BAD_TUPLE;
  1145.     org->data_org = *p;
  1146.     if (++p == q) return CS_BAD_TUPLE;
  1147.     for (i = 0; i < 30; i++) {
  1148. org->desc[i] = *p;
  1149. if (*p == '') break;
  1150. if (++p == q) return CS_BAD_TUPLE;
  1151.     }
  1152.     return CS_SUCCESS;
  1153. }
  1154. /*====================================================================*/
  1155. static int parse_format(tuple_t *tuple, cistpl_format_t *fmt)
  1156. {
  1157.     u_char *p;
  1158.     if (tuple->TupleDataLen < 10)
  1159. return CS_BAD_TUPLE;
  1160.     p = tuple->TupleData;
  1161.     fmt->type = p[0];
  1162.     fmt->edc = p[1];
  1163.     fmt->offset = le32_to_cpu(*(u_int *)(p+2));
  1164.     fmt->length = le32_to_cpu(*(u_int *)(p+6));
  1165.     return CS_SUCCESS;
  1166. }
  1167. /*====================================================================*/
  1168. int pcmcia_parse_tuple(client_handle_t handle, tuple_t *tuple, cisparse_t *parse)
  1169. {
  1170.     int ret = CS_SUCCESS;
  1171.     
  1172.     if (tuple->TupleDataLen > tuple->TupleDataMax)
  1173. return CS_BAD_TUPLE;
  1174.     switch (tuple->TupleCode) {
  1175.     case CISTPL_DEVICE:
  1176.     case CISTPL_DEVICE_A:
  1177. ret = parse_device(tuple, &parse->device);
  1178. break;
  1179. #ifdef CONFIG_CARDBUS
  1180.     case CISTPL_BAR:
  1181. ret = parse_bar(tuple, &parse->bar);
  1182. break;
  1183.     case CISTPL_CONFIG_CB:
  1184. ret = parse_config_cb(tuple, &parse->config);
  1185. break;
  1186.     case CISTPL_CFTABLE_ENTRY_CB:
  1187. ret = parse_cftable_entry_cb(tuple, &parse->cftable_entry_cb);
  1188. break;
  1189. #endif
  1190.     case CISTPL_CHECKSUM:
  1191. ret = parse_checksum(tuple, &parse->checksum);
  1192. break;
  1193.     case CISTPL_LONGLINK_A:
  1194.     case CISTPL_LONGLINK_C:
  1195. ret = parse_longlink(tuple, &parse->longlink);
  1196. break;
  1197.     case CISTPL_LONGLINK_MFC:
  1198. ret = parse_longlink_mfc(tuple, &parse->longlink_mfc);
  1199. break;
  1200.     case CISTPL_VERS_1:
  1201. ret = parse_vers_1(tuple, &parse->version_1);
  1202. break;
  1203.     case CISTPL_ALTSTR:
  1204. ret = parse_altstr(tuple, &parse->altstr);
  1205. break;
  1206.     case CISTPL_JEDEC_A:
  1207.     case CISTPL_JEDEC_C:
  1208. ret = parse_jedec(tuple, &parse->jedec);
  1209. break;
  1210.     case CISTPL_MANFID:
  1211. ret = parse_manfid(tuple, &parse->manfid);
  1212. break;
  1213.     case CISTPL_FUNCID:
  1214. ret = parse_funcid(tuple, &parse->funcid);
  1215. break;
  1216.     case CISTPL_FUNCE:
  1217. ret = parse_funce(tuple, &parse->funce);
  1218. break;
  1219.     case CISTPL_CONFIG:
  1220. ret = parse_config(tuple, &parse->config);
  1221. break;
  1222.     case CISTPL_CFTABLE_ENTRY:
  1223. ret = parse_cftable_entry(tuple, &parse->cftable_entry);
  1224. break;
  1225.     case CISTPL_DEVICE_GEO:
  1226.     case CISTPL_DEVICE_GEO_A:
  1227. ret = parse_device_geo(tuple, &parse->device_geo);
  1228. break;
  1229.     case CISTPL_VERS_2:
  1230. ret = parse_vers_2(tuple, &parse->vers_2);
  1231. break;
  1232.     case CISTPL_ORG:
  1233. ret = parse_org(tuple, &parse->org);
  1234. break;
  1235.     case CISTPL_FORMAT:
  1236.     case CISTPL_FORMAT_A:
  1237. ret = parse_format(tuple, &parse->format);
  1238. break;
  1239.     case CISTPL_NO_LINK:
  1240.     case CISTPL_LINKTARGET:
  1241. ret = CS_SUCCESS;
  1242. break;
  1243.     default:
  1244. ret = CS_UNSUPPORTED_FUNCTION;
  1245. break;
  1246.     }
  1247.     return ret;
  1248. }
  1249. /*======================================================================
  1250.     This is used internally by Card Services to look up CIS stuff.
  1251.     
  1252. ======================================================================*/
  1253. int read_tuple(client_handle_t handle, cisdata_t code, void *parse)
  1254. {
  1255.     tuple_t tuple;
  1256.     cisdata_t buf[255];
  1257.     int ret;
  1258.     
  1259.     tuple.DesiredTuple = code;
  1260.     tuple.Attributes = TUPLE_RETURN_COMMON;
  1261.     ret = pcmcia_get_first_tuple(handle, &tuple);
  1262.     if (ret != CS_SUCCESS) return ret;
  1263.     tuple.TupleData = buf;
  1264.     tuple.TupleOffset = 0;
  1265.     tuple.TupleDataMax = sizeof(buf);
  1266.     ret = pcmcia_get_tuple_data(handle, &tuple);
  1267.     if (ret != CS_SUCCESS) return ret;
  1268.     ret = pcmcia_parse_tuple(handle, &tuple, parse);
  1269.     return ret;
  1270. }
  1271. /*======================================================================
  1272.     This tries to determine if a card has a sensible CIS.  It returns
  1273.     the number of tuples in the CIS, or 0 if the CIS looks bad.  The
  1274.     checks include making sure several critical tuples are present and
  1275.     valid; seeing if the total number of tuples is reasonable; and
  1276.     looking for tuples that use reserved codes.
  1277.     
  1278. ======================================================================*/
  1279. int pcmcia_validate_cis(client_handle_t handle, cisinfo_t *info)
  1280. {
  1281.     tuple_t tuple;
  1282.     cisparse_t p;
  1283.     int ret, reserved, dev_ok = 0, ident_ok = 0;
  1284.     if (CHECK_HANDLE(handle))
  1285. return CS_BAD_HANDLE;
  1286.     info->Chains = reserved = 0;
  1287.     tuple.DesiredTuple = RETURN_FIRST_TUPLE;
  1288.     tuple.Attributes = TUPLE_RETURN_COMMON;
  1289.     ret = pcmcia_get_first_tuple(handle, &tuple);
  1290.     if (ret != CS_SUCCESS)
  1291. return CS_SUCCESS;
  1292.     /* First tuple should be DEVICE; we should really have either that
  1293.        or a CFTABLE_ENTRY of some sort */
  1294.     if ((tuple.TupleCode == CISTPL_DEVICE) ||
  1295. (read_tuple(handle, CISTPL_CFTABLE_ENTRY, &p) == CS_SUCCESS) ||
  1296. (read_tuple(handle, CISTPL_CFTABLE_ENTRY_CB, &p) == CS_SUCCESS))
  1297. dev_ok++;
  1298.     /* All cards should have a MANFID tuple, and/or a VERS_1 or VERS_2
  1299.        tuple, for card identification.  Certain old D-Link and Linksys
  1300.        cards have only a broken VERS_2 tuple; hence the bogus test. */
  1301.     if ((read_tuple(handle, CISTPL_MANFID, &p) == CS_SUCCESS) ||
  1302. (read_tuple(handle, CISTPL_VERS_1, &p) == CS_SUCCESS) ||
  1303. (read_tuple(handle, CISTPL_VERS_2, &p) != CS_NO_MORE_ITEMS))
  1304. ident_ok++;
  1305.     if (!dev_ok && !ident_ok)
  1306. return CS_SUCCESS;
  1307.     for (info->Chains = 1; info->Chains < MAX_TUPLES; info->Chains++) {
  1308. ret = pcmcia_get_next_tuple(handle, &tuple);
  1309. if (ret != CS_SUCCESS) break;
  1310. if (((tuple.TupleCode > 0x23) && (tuple.TupleCode < 0x40)) ||
  1311.     ((tuple.TupleCode > 0x47) && (tuple.TupleCode < 0x80)) ||
  1312.     ((tuple.TupleCode > 0x90) && (tuple.TupleCode < 0xff)))
  1313.     reserved++;
  1314.     }
  1315.     if ((info->Chains == MAX_TUPLES) || (reserved > 5) ||
  1316. ((!dev_ok || !ident_ok) && (info->Chains > 10)))
  1317. info->Chains = 0;
  1318.     return CS_SUCCESS;
  1319. }