aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/char/pcmcia/ipwireless/hardware.c51
1 files changed, 38 insertions, 13 deletions
diff --git a/drivers/char/pcmcia/ipwireless/hardware.c b/drivers/char/pcmcia/ipwireless/hardware.c
index 7428734d08a1..08423ba5b9dd 100644
--- a/drivers/char/pcmcia/ipwireless/hardware.c
+++ b/drivers/char/pcmcia/ipwireless/hardware.c
@@ -132,29 +132,17 @@ enum {
132#define NL_FOLLOWING_PACKET_HEADER_SIZE 1 132#define NL_FOLLOWING_PACKET_HEADER_SIZE 1
133 133
134struct nl_first_packet_header { 134struct nl_first_packet_header {
135#if defined(__BIG_ENDIAN_BITFIELD)
136 unsigned char packet_rank:2;
137 unsigned char address:3;
138 unsigned char protocol:3;
139#else
140 unsigned char protocol:3; 135 unsigned char protocol:3;
141 unsigned char address:3; 136 unsigned char address:3;
142 unsigned char packet_rank:2; 137 unsigned char packet_rank:2;
143#endif
144 unsigned char length_lsb; 138 unsigned char length_lsb;
145 unsigned char length_msb; 139 unsigned char length_msb;
146}; 140};
147 141
148struct nl_packet_header { 142struct nl_packet_header {
149#if defined(__BIG_ENDIAN_BITFIELD)
150 unsigned char packet_rank:2;
151 unsigned char address:3;
152 unsigned char protocol:3;
153#else
154 unsigned char protocol:3; 143 unsigned char protocol:3;
155 unsigned char address:3; 144 unsigned char address:3;
156 unsigned char packet_rank:2; 145 unsigned char packet_rank:2;
157#endif
158}; 146};
159 147
160/* Value of 'packet_rank' above */ 148/* Value of 'packet_rank' above */
@@ -382,7 +370,37 @@ static void dump_data_bytes(const char *type, const unsigned char *data,
382 length < DUMP_MAX_BYTES ? length : DUMP_MAX_BYTES); 370 length < DUMP_MAX_BYTES ? length : DUMP_MAX_BYTES);
383} 371}
384 372
385static int do_send_fragment(struct ipw_hardware *hw, const unsigned char *data, 373static void swap_packet_bitfield_to_le(unsigned char *data)
374{
375#ifdef __BIG_ENDIAN_BITFIELD
376 unsigned char tmp = *data, ret = 0;
377
378 /*
379 * transform bits from aa.bbb.ccc to ccc.bbb.aa
380 */
381 ret |= tmp & 0xc0 >> 6;
382 ret |= tmp & 0x38 >> 1;
383 ret |= tmp & 0x07 << 5;
384 *data = ret & 0xff;
385#endif
386}
387
388static void swap_packet_bitfield_from_le(unsigned char *data)
389{
390#ifdef __BIG_ENDIAN_BITFIELD
391 unsigned char tmp = *data, ret = 0;
392
393 /*
394 * transform bits from ccc.bbb.aa to aa.bbb.ccc
395 */
396 ret |= tmp & 0xe0 >> 5;
397 ret |= tmp & 0x1c << 1;
398 ret |= tmp & 0x03 << 6;
399 *data = ret & 0xff;
400#endif
401}
402
403static int do_send_fragment(struct ipw_hardware *hw, unsigned char *data,
386 unsigned length) 404 unsigned length)
387{ 405{
388 unsigned i; 406 unsigned i;
@@ -402,6 +420,7 @@ static int do_send_fragment(struct ipw_hardware *hw, const unsigned char *data,
402 spin_lock_irqsave(&hw->lock, flags); 420 spin_lock_irqsave(&hw->lock, flags);
403 421
404 hw->tx_ready = 0; 422 hw->tx_ready = 0;
423 swap_packet_bitfield_to_le(data);
405 424
406 if (hw->hw_version == HW_VERSION_1) { 425 if (hw->hw_version == HW_VERSION_1) {
407 outw((unsigned short) length, hw->base_port + IODWR); 426 outw((unsigned short) length, hw->base_port + IODWR);
@@ -458,6 +477,10 @@ static int do_send_packet(struct ipw_hardware *hw, struct ipw_tx_packet *packet)
458 if (data_left < fragment_data_len) 477 if (data_left < fragment_data_len)
459 fragment_data_len = data_left; 478 fragment_data_len = data_left;
460 479
480 /*
481 * hdr_first is now in machine bitfield order, which will be swapped
482 * to le just before it goes to hw
483 */
461 pkt.hdr_first.protocol = packet->protocol; 484 pkt.hdr_first.protocol = packet->protocol;
462 pkt.hdr_first.address = packet->dest_addr; 485 pkt.hdr_first.address = packet->dest_addr;
463 pkt.hdr_first.packet_rank = 0; 486 pkt.hdr_first.packet_rank = 0;
@@ -883,6 +906,8 @@ static void do_receive_packet(struct ipw_hardware *hw)
883 906
884 acknowledge_data_read(hw); 907 acknowledge_data_read(hw);
885 908
909 swap_packet_bitfield_from_le(pkt);
910
886 if (ipwireless_debug) 911 if (ipwireless_debug)
887 dump_data_bytes("recv", pkt, len); 912 dump_data_bytes("recv", pkt, len);
888 913