Commit 37ace3d4 authored by Dominik Brodowski's avatar Dominik Brodowski

pcmcia: convert ssb pcmcia driver to use new CIS helpers

SSB is a prime example of how to make use of the new CIS helpers.

CC: Michael Buesch <mb@bu3sch.de>
Acked-by: default avatarJohn W. Linville <linville@tuxdriver.com>
Signed-off-by: default avatarDominik Brodowski <linux@dominikbrodowski.net>
parent dddfbd82
...@@ -617,132 +617,104 @@ static int ssb_pcmcia_sprom_check_crc(const u16 *sprom, size_t size) ...@@ -617,132 +617,104 @@ static int ssb_pcmcia_sprom_check_crc(const u16 *sprom, size_t size)
} \ } \
} while (0) } while (0)
int ssb_pcmcia_get_invariants(struct ssb_bus *bus, static int ssb_pcmcia_get_mac(struct pcmcia_device *p_dev,
struct ssb_init_invariants *iv) tuple_t *tuple,
void *priv)
{ {
tuple_t tuple; struct ssb_sprom *sprom = priv;
int res;
unsigned char buf[32]; if (tuple->TupleData[0] != CISTPL_FUNCE_LAN_NODE_ID)
return -EINVAL;
if (tuple->TupleDataLen != ETH_ALEN + 2)
return -EINVAL;
if (tuple->TupleData[1] != ETH_ALEN)
return -EINVAL;
memcpy(sprom->il0mac, &tuple->TupleData[2], ETH_ALEN);
return 0;
};
static int ssb_pcmcia_do_get_invariants(struct pcmcia_device *p_dev,
tuple_t *tuple,
void *priv)
{
struct ssb_init_invariants *iv = priv;
struct ssb_sprom *sprom = &iv->sprom; struct ssb_sprom *sprom = &iv->sprom;
struct ssb_boardinfo *bi = &iv->boardinfo; struct ssb_boardinfo *bi = &iv->boardinfo;
const char *error_description; const char *error_description;
memset(sprom, 0xFF, sizeof(*sprom)); GOTO_ERROR_ON(tuple->TupleDataLen < 1, "VEN tpl < 1");
sprom->revision = 1; switch (tuple->TupleData[0]) {
sprom->boardflags_lo = 0;
sprom->boardflags_hi = 0;
/* First fetch the MAC address. */
memset(&tuple, 0, sizeof(tuple));
tuple.DesiredTuple = CISTPL_FUNCE;
tuple.TupleData = buf;
tuple.TupleDataMax = sizeof(buf);
res = pcmcia_get_first_tuple(bus->host_pcmcia, &tuple);
GOTO_ERROR_ON(res != 0, "MAC first tpl");
res = pcmcia_get_tuple_data(bus->host_pcmcia, &tuple);
GOTO_ERROR_ON(res != 0, "MAC first tpl data");
while (1) {
GOTO_ERROR_ON(tuple.TupleDataLen < 1, "MAC tpl < 1");
if (tuple.TupleData[0] == CISTPL_FUNCE_LAN_NODE_ID)
break;
res = pcmcia_get_next_tuple(bus->host_pcmcia, &tuple);
GOTO_ERROR_ON(res != 0, "MAC next tpl");
res = pcmcia_get_tuple_data(bus->host_pcmcia, &tuple);
GOTO_ERROR_ON(res != 0, "MAC next tpl data");
}
GOTO_ERROR_ON(tuple.TupleDataLen != ETH_ALEN + 2, "MAC tpl size");
memcpy(sprom->il0mac, &tuple.TupleData[2], ETH_ALEN);
/* Fetch the vendor specific tuples. */
memset(&tuple, 0, sizeof(tuple));
tuple.DesiredTuple = SSB_PCMCIA_CIS;
tuple.TupleData = buf;
tuple.TupleDataMax = sizeof(buf);
res = pcmcia_get_first_tuple(bus->host_pcmcia, &tuple);
GOTO_ERROR_ON(res != 0, "VEN first tpl");
res = pcmcia_get_tuple_data(bus->host_pcmcia, &tuple);
GOTO_ERROR_ON(res != 0, "VEN first tpl data");
while (1) {
GOTO_ERROR_ON(tuple.TupleDataLen < 1, "VEN tpl < 1");
switch (tuple.TupleData[0]) {
case SSB_PCMCIA_CIS_ID: case SSB_PCMCIA_CIS_ID:
GOTO_ERROR_ON((tuple.TupleDataLen != 5) && GOTO_ERROR_ON((tuple->TupleDataLen != 5) &&
(tuple.TupleDataLen != 7), (tuple->TupleDataLen != 7),
"id tpl size"); "id tpl size");
bi->vendor = tuple.TupleData[1] | bi->vendor = tuple->TupleData[1] |
((u16)tuple.TupleData[2] << 8); ((u16)tuple->TupleData[2] << 8);
break; break;
case SSB_PCMCIA_CIS_BOARDREV: case SSB_PCMCIA_CIS_BOARDREV:
GOTO_ERROR_ON(tuple.TupleDataLen != 2, GOTO_ERROR_ON(tuple->TupleDataLen != 2,
"boardrev tpl size"); "boardrev tpl size");
sprom->board_rev = tuple.TupleData[1]; sprom->board_rev = tuple->TupleData[1];
break; break;
case SSB_PCMCIA_CIS_PA: case SSB_PCMCIA_CIS_PA:
GOTO_ERROR_ON((tuple.TupleDataLen != 9) && GOTO_ERROR_ON((tuple->TupleDataLen != 9) &&
(tuple.TupleDataLen != 10), (tuple->TupleDataLen != 10),
"pa tpl size"); "pa tpl size");
sprom->pa0b0 = tuple.TupleData[1] | sprom->pa0b0 = tuple->TupleData[1] |
((u16)tuple.TupleData[2] << 8); ((u16)tuple->TupleData[2] << 8);
sprom->pa0b1 = tuple.TupleData[3] | sprom->pa0b1 = tuple->TupleData[3] |
((u16)tuple.TupleData[4] << 8); ((u16)tuple->TupleData[4] << 8);
sprom->pa0b2 = tuple.TupleData[5] | sprom->pa0b2 = tuple->TupleData[5] |
((u16)tuple.TupleData[6] << 8); ((u16)tuple->TupleData[6] << 8);
sprom->itssi_a = tuple.TupleData[7]; sprom->itssi_a = tuple->TupleData[7];
sprom->itssi_bg = tuple.TupleData[7]; sprom->itssi_bg = tuple->TupleData[7];
sprom->maxpwr_a = tuple.TupleData[8]; sprom->maxpwr_a = tuple->TupleData[8];
sprom->maxpwr_bg = tuple.TupleData[8]; sprom->maxpwr_bg = tuple->TupleData[8];
break; break;
case SSB_PCMCIA_CIS_OEMNAME: case SSB_PCMCIA_CIS_OEMNAME:
/* We ignore this. */ /* We ignore this. */
break; break;
case SSB_PCMCIA_CIS_CCODE: case SSB_PCMCIA_CIS_CCODE:
GOTO_ERROR_ON(tuple.TupleDataLen != 2, GOTO_ERROR_ON(tuple->TupleDataLen != 2,
"ccode tpl size"); "ccode tpl size");
sprom->country_code = tuple.TupleData[1]; sprom->country_code = tuple->TupleData[1];
break; break;
case SSB_PCMCIA_CIS_ANTENNA: case SSB_PCMCIA_CIS_ANTENNA:
GOTO_ERROR_ON(tuple.TupleDataLen != 2, GOTO_ERROR_ON(tuple->TupleDataLen != 2,
"ant tpl size"); "ant tpl size");
sprom->ant_available_a = tuple.TupleData[1]; sprom->ant_available_a = tuple->TupleData[1];
sprom->ant_available_bg = tuple.TupleData[1]; sprom->ant_available_bg = tuple->TupleData[1];
break; break;
case SSB_PCMCIA_CIS_ANTGAIN: case SSB_PCMCIA_CIS_ANTGAIN:
GOTO_ERROR_ON(tuple.TupleDataLen != 2, GOTO_ERROR_ON(tuple->TupleDataLen != 2,
"antg tpl size"); "antg tpl size");
sprom->antenna_gain.ghz24.a0 = tuple.TupleData[1]; sprom->antenna_gain.ghz24.a0 = tuple->TupleData[1];
sprom->antenna_gain.ghz24.a1 = tuple.TupleData[1]; sprom->antenna_gain.ghz24.a1 = tuple->TupleData[1];
sprom->antenna_gain.ghz24.a2 = tuple.TupleData[1]; sprom->antenna_gain.ghz24.a2 = tuple->TupleData[1];
sprom->antenna_gain.ghz24.a3 = tuple.TupleData[1]; sprom->antenna_gain.ghz24.a3 = tuple->TupleData[1];
sprom->antenna_gain.ghz5.a0 = tuple.TupleData[1]; sprom->antenna_gain.ghz5.a0 = tuple->TupleData[1];
sprom->antenna_gain.ghz5.a1 = tuple.TupleData[1]; sprom->antenna_gain.ghz5.a1 = tuple->TupleData[1];
sprom->antenna_gain.ghz5.a2 = tuple.TupleData[1]; sprom->antenna_gain.ghz5.a2 = tuple->TupleData[1];
sprom->antenna_gain.ghz5.a3 = tuple.TupleData[1]; sprom->antenna_gain.ghz5.a3 = tuple->TupleData[1];
break; break;
case SSB_PCMCIA_CIS_BFLAGS: case SSB_PCMCIA_CIS_BFLAGS:
GOTO_ERROR_ON((tuple.TupleDataLen != 3) && GOTO_ERROR_ON((tuple->TupleDataLen != 3) &&
(tuple.TupleDataLen != 5), (tuple->TupleDataLen != 5),
"bfl tpl size"); "bfl tpl size");
sprom->boardflags_lo = tuple.TupleData[1] | sprom->boardflags_lo = tuple->TupleData[1] |
((u16)tuple.TupleData[2] << 8); ((u16)tuple->TupleData[2] << 8);
break; break;
case SSB_PCMCIA_CIS_LEDS: case SSB_PCMCIA_CIS_LEDS:
GOTO_ERROR_ON(tuple.TupleDataLen != 5, GOTO_ERROR_ON(tuple->TupleDataLen != 5,
"leds tpl size"); "leds tpl size");
sprom->gpio0 = tuple.TupleData[1]; sprom->gpio0 = tuple->TupleData[1];
sprom->gpio1 = tuple.TupleData[2]; sprom->gpio1 = tuple->TupleData[2];
sprom->gpio2 = tuple.TupleData[3]; sprom->gpio2 = tuple->TupleData[3];
sprom->gpio3 = tuple.TupleData[4]; sprom->gpio3 = tuple->TupleData[4];
break;
}
res = pcmcia_get_next_tuple(bus->host_pcmcia, &tuple);
if (res == -ENOSPC)
break; break;
GOTO_ERROR_ON(res != 0, "VEN next tpl");
res = pcmcia_get_tuple_data(bus->host_pcmcia, &tuple);
GOTO_ERROR_ON(res != 0, "VEN next tpl data");
} }
return -ENOSPC; /* continue with next entry */
return 0;
error: error:
ssb_printk(KERN_ERR PFX ssb_printk(KERN_ERR PFX
"PCMCIA: Failed to fetch device invariants: %s\n", "PCMCIA: Failed to fetch device invariants: %s\n",
...@@ -750,6 +722,38 @@ error: ...@@ -750,6 +722,38 @@ error:
return -ENODEV; return -ENODEV;
} }
int ssb_pcmcia_get_invariants(struct ssb_bus *bus,
struct ssb_init_invariants *iv)
{
struct ssb_sprom *sprom = &iv->sprom;
int res;
memset(sprom, 0xFF, sizeof(*sprom));
sprom->revision = 1;
sprom->boardflags_lo = 0;
sprom->boardflags_hi = 0;
/* First fetch the MAC address. */
res = pcmcia_loop_tuple(bus->host_pcmcia, CISTPL_FUNCE,
ssb_pcmcia_get_mac, sprom);
if (res != 0) {
ssb_printk(KERN_ERR PFX
"PCMCIA: Failed to fetch MAC address\n");
return -ENODEV;
}
/* Fetch the vendor specific tuples. */
res = pcmcia_loop_tuple(bus->host_pcmcia, SSB_PCMCIA_CIS,
ssb_pcmcia_do_get_invariants, sprom);
if ((res == 0) || (res == -ENOSPC))
return 0;
ssb_printk(KERN_ERR PFX
"PCMCIA: Failed to fetch device invariants\n");
return -ENODEV;
}
static ssize_t ssb_pcmcia_attr_sprom_show(struct device *pcmciadev, static ssize_t ssb_pcmcia_attr_sprom_show(struct device *pcmciadev,
struct device_attribute *attr, struct device_attribute *attr,
char *buf) char *buf)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment