/*====================================================================== A utility to convert a plain text description of a Card Information Structure into its packed binary representation. pack_cis.c 1.20 2002/10/16 16:38:18 The contents of this file are subject to the Mozilla Public License Version 1.1 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.mozilla.org/MPL/ Software distributed under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the specific language governing rights and limitations under the License. The initial developer of the original code is David A. Hinds . Portions created by David A. Hinds are Copyright (C) 1999 David A. Hinds. All Rights Reserved. Alternatively, the contents of this file may be used under the terms of the GNU General Public License version 2 (the "GPL"), in which case the provisions of the GPL are applicable instead of the above. If you wish to allow the use of your version of this file only under the terms of the GPL and not to allow others to use your version of this file under the MPL, indicate your decision by deleting the provisions above and replace them with the notice and other provisions required by the GPL. If you do not delete the provisions above, a recipient may use your version of this file under either the MPL or the GPL. Usage: pack_cis [-o outfile] [infile] [infile] defaults to stdin, and [outfile] defaults to stdout. ======================================================================*/ #include #include #include #include #include #include #include #include #include #include "pack_cis.h" tuple_info_t *cis_root = NULL, *mfc[8] = { NULL }; u_int nf = 0; /*====================================================================== Support routines for packing parts of configuration table entries ======================================================================*/ static u_int mantissa[] = { 10, 12, 13, 15, 20, 25, 30, 35, 40, 45, 50, 55, 60, 70, 80, 90 }; static int pack_power(cistpl_power_t *pwr, u_char *b) { u_int tmp, i; u_char m, e, x, *c = b; *c = pwr->present; c++; for (i = 0; i < 7; i++) { if (!(pwr->present & (1<param[i]; for (e = 1; ((tmp % 10) == 0) || (tmp > 999); e++) tmp /= 10; x = m = 0; if (tmp < 100) { if (tmp < 10) { tmp *= 10; e--; } for (m = 0; m < 16; m++) if (mantissa[m] == tmp) break; if (m == 16) { tmp *= 10; e--; } } if (tmp >= 100) { e++; x = (tmp/10) - ((tmp/10) % 10); for (m = 0; m < 16; m++) if (mantissa[m] == x) break; x = (u_char)(tmp - 10*(u_int)x); } *c = (m<<3) | e | (x ? 0x80 : 0); c++; if (x) { *c = x; c++; } } return c-b; } static int pack_io(cistpl_io_t *p, u_char *b) { u_char *c = b; u_int i, j, ml, ma; *c = p->flags & (CISTPL_IO_8BIT|CISTPL_IO_16BIT); if ((p->nwin == 1) && (p->win[0].base == 0)) { for (i = 1, j = 0; i < p->win[0].len; i *= 2, j++) ; *c |= j; c++; } else { for (i = ma = ml = 0; i < p->nwin; i++) { ma |= p->win[i].base; ml |= p->win[i].len-1; } ma = (ma > 0xffff) ? 3 : ((ma > 0xff) ? 2 : 1); ml = (ml > 0xffff) ? 3 : ((ml > 0xff) ? 2 : 1); *c |= 0x80 | (p->flags & CISTPL_IO_LINES_MASK); c++; *c = (p->nwin-1) | (ma<<4) | (ml<<6); c++; if (ma == 3) ma++; if (ml == 3) ml++; for (i = 0; i < p->nwin; i++) { for (j = 0; j < ma; j++) { *c = (p->win[i].base >> (8*j)) & 0xff; c++; } for (j = 0; j < ml; j++) { *c = ((p->win[i].len-1) >> (8*j)) & 0xff; c++; } } } return c-b; } static int pack_mem(cistpl_mem_t *p, u_char *b) { u_char *c = b; u_int i, j, ml, ma, ha; for (i = ma = ml = ha = 0; i < p->nwin; i++) { ma |= p->win[i].card_addr; ml |= p->win[i].len; ha |= p->win[i].host_addr; } ma = (ma|ha) >> 8; ml >>= 8; ma = (ma > 0xffff) ? 3 : ((ma > 0xff) ? 2 : 1); ml = (ml > 0xffff) ? 3 : ((ml > 0xff) ? 2 : 1); *c = (p->nwin-1) | (ma<<5) | (ml<<3) | (ha ? 0x80 : 0); c++; for (i = 0; i < p->nwin; i++) { for (j = 1; j <= ml; j++) { *c = (p->win[i].len >> (8*j)) & 0xff; c++; } for (j = 1; j <= ma; j++) { *c = (p->win[i].card_addr >> (8*j)) & 0xff; c++; } if (ha) for (j = 1; j <= ma; j++) { *c = (p->win[i].host_addr >> (8*j)) & 0xff; c++; } } return c-b; } static int pack_irq(cistpl_irq_t *p, u_char *b) { b[0] = p->IRQInfo1; if (p->IRQInfo1 & IRQ_INFO2_VALID) { b[1] = p->IRQInfo2 & 0xff; b[2] = (p->IRQInfo2 >> 8) & 0xff; return 3; } return 1; } static void pack_cftable(cistpl_cftable_entry_t *p, u_char *b) { u_char *c; b[2] = p->index | 0x80; if (p->flags & CISTPL_CFTABLE_DEFAULT) b[2] |= 0x40; b[3] = 0x01; b[3] |= (p->flags & CISTPL_CFTABLE_BVDS) ? 0x10 : 0; b[3] |= (p->flags & CISTPL_CFTABLE_WP) ? 0x20 : 0; b[3] |= (p->flags & CISTPL_CFTABLE_RDYBSY) ? 0x40 : 0; b[3] |= (p->flags & CISTPL_CFTABLE_MWAIT) ? 0x80 : 0; b[4] = 0; c = b+5; if (p->vcc.present) { b[4]++; c += pack_power(&p->vcc, c); if (p->vpp1.present) { b[4]++; c += pack_power(&p->vpp1, c); if (p->vpp2.present) { b[4]++; c += pack_power(&p->vpp2, c); } } } if (p->io.nwin > 0) { b[4] |= 0x08; c += pack_io(&p->io, c); } if (p->irq.IRQInfo1 > 0) { b[4] |= 0x10; c += pack_irq(&p->irq, c); } if (p->mem.nwin > 0) { b[4] |= 0x60; c += pack_mem(&p->mem, c); } if (p->flags >> 8) { b[4] |= 0x80; *c++ = p->flags >> 8; } b[1] = c-b-2; } /*====================================================================== Routines for packing device info tuples ======================================================================*/ static int pack_speed(u_int speed, u_char *b) { u_char e, m, *c = b; switch (speed) { case 0: *c |= 0; c++; break; case 250: *c |= 1; c++; break; case 200: *c |= 2; c++; break; case 150: *c |= 3; c++; break; case 100: *c |= 4; c++; break; default: *c |= 7; c++; for (e = 1; speed > 80; e++) speed /= 10; for (m = 0; m < 15; m++) if (mantissa[m] >= speed) break; *c = ((m+1)<<3) | e; c++; } return c-b; } static void pack_device(cistpl_device_t *d, u_char *b) { u_int i, sz; u_char e, *c = b+2; for (i = 0; i < d->ndev; i++) { *c = (d->dev[i].type<<4); c += pack_speed(d->dev[i].speed, c); sz = d->dev[i].size/512; for (e = 0; sz > 32; e++) sz /= 4; *c = (e & 7) | ((sz-1) << 3); c++; } *c = 0xff; c++; b[1] = c-b-2; } /*====================================================================== For now, I only implement a subset of tuples types, intended to be enough to handle most IO-oriented cards. ======================================================================*/ static int pack_tuple(tuple_info_t *t, u_char *b) { cisparse_t *p = t->parse; u_int i, m; u_char *c; *b = t->type; switch (t->type) { case CISTPL_DEVICE: case CISTPL_DEVICE_A: if (p) { pack_device(&p->device, b); } else { /* Fake null device tuple */ b[1] = 3; b[2] = 0; b[3] = 0; b[4] = 0xff; } break; case CISTPL_MANFID: b[1] = 4; b[2] = p->manfid.manf & 0xff; b[3] = p->manfid.manf >> 8; b[4] = p->manfid.card & 0xff; b[5] = p->manfid.card >> 8; break; case CISTPL_FUNCID: b[1] = 2; b[2] = p->funcid.func; b[3] = p->funcid.sysinit; break; case CISTPL_JEDEC_C: case CISTPL_JEDEC_A: b[1] = 2*p->jedec.nid; for (i = 0; i < p->jedec.nid; i++) { b[2*i+1] = p->jedec.id[i].mfr; b[2*i+2] = p->jedec.id[i].info; } break; case CISTPL_CONFIG: b[3] = p->config.last_idx; i = p->config.base; for (c = b+4, m = 0; (i > 0) || !m; i >>= 8, m++) { c[m] = i & 0xff; } b[2] = m-1; i = p->config.rmask[0]; for (c = c+m, m = 0; (i > 0) || !m; i >>= 8, m++) { c[m] = i & 0xff; } b[2] |= ((m-1) << 2); b[1] = c+m-b-2; break; case CISTPL_VERS_1: b[2] = p->version_1.major; b[3] = p->version_1.minor; c = b+4; for (i = 0; i < p->version_1.ns; i++) { strcpy((char *)c, p->version_1.str+p->version_1.ofs[i]); c += strlen((char *)c) + 1; } for (; i < 4; i++) { *c = 0; c++; } *c = 0xff; c++; b[1] = c-b-2; break; case CISTPL_CFTABLE_ENTRY: pack_cftable(&p->cftable_entry, b); break; case CISTPL_LINKTARGET: b[1] = 3; b[2] = 'C'; b[3] = 'I'; b[4] = 'S'; break; case CISTPL_NO_LINK: case CISTPL_END: b[1] = 0; break; } return b[1]+2; } /*====================================================================== The following routines handle parsing of aggregates of tuples. pack_chain() is the simplest: just return a string of tuples and terminate with an END tuple. pack_mfc() is used to tie the function-specific tuple chains for a multifunction card together using a LONGLINK_MFC tuple. And pack_cis() handles a complete CIS, whether it is multifunction or not. ======================================================================*/ static int pack_chain(tuple_info_t *t, u_char *b) { int n = 0; tuple_info_t end = { CISTPL_END, NULL, NULL }; while (t) { n += pack_tuple(t, b+n); t = t->next; } n += pack_tuple(&end, b+n); return n; } static int pack_mfc(u_int ofs, u_char *b) { u_int i, j, pos; tuple_info_t target = { CISTPL_LINKTARGET, NULL, NULL }; b[0] = CISTPL_LONGLINK_MFC; b[1] = 5*nf + 1; b[2] = nf; b[5*nf+3] = CISTPL_END; b[5*nf+4] = 0; /* Leave space for this tuple and the CISTPL_END tuple */ pos = 5*nf+5; for (i = 0; i < nf; i++) { b[3+i*5] = 0; for (j = 0; j < 4; j++) b[4+i*5+j] = ((ofs+pos) >> (8*j)) & 0xff; pos += pack_tuple(&target, b+pos); pos += pack_chain(mfc[i], b+pos); } return ofs+pos; } static int pack_cis(tuple_info_t *t, u_char *b) { int n = 0; tuple_info_t device = { CISTPL_DEVICE, NULL, NULL }; tuple_info_t nolink = { CISTPL_NO_LINK, NULL, NULL }; tuple_info_t end = { CISTPL_END, NULL, NULL }; if (t->type != CISTPL_DEVICE) n = pack_tuple(&device, b); while (t) { n += pack_tuple(t, b+n); t = t->next; } if (nf > 0) { n = pack_mfc(n, b+n); } else { n += pack_tuple(&nolink, b+n); n += pack_tuple(&end, b+n); } return n; } /*====================================================================*/ int main(int argc, char *argv[]) { int optch, errflg = 0; char *out = NULL; u_char buf[1024]; int n; FILE *f; while ((optch = getopt(argc, argv, "o:")) != -1) { switch (optch) { case 'o': out = strdup(optarg); break; default: errflg = 1; break; } } if (errflg || (optind < argc-1)) { fprintf(stderr, "usage: %s [-o outfile] [infile]\n", argv[0]); exit(EXIT_FAILURE); } if (optind < argc) { f = fopen(argv[optind], "r"); if (!f) { fprintf(stderr, "could not open '%s': %s\n", argv[optind], strerror(errno)); return -1; } } else f = stdin; parse_cis(f); fclose(f); n = pack_cis(cis_root, buf); if (out) { f = fopen(out, "w"); if (!f) { fprintf(stderr, "could not open '%s': %s\n", out, strerror(errno)); return -1; } } else f = stdout; fwrite(buf, n, 1, f); fclose(f); return 0; }