rt2x00: Driver requiring firmware should select crc algo
authorIvo van Doorn <ivdoorn@gmail.com>
Sun, 3 Feb 2008 14:48:38 +0000 (15:48 +0100)
committerJohn W. Linville <linville@tuxdriver.com>
Fri, 29 Feb 2008 20:19:28 +0000 (15:19 -0500)
The driver should select what CRC algorithm is required
when performing a checksum on the firmware.

rt61pci & rt73usb require crc-itu-t
rt2800pci & rt2800usb require crc-ccitt

Legacy 2800pci/usb driver uses crc-itu-t + bit order reversion,
but that is just inefficient especially since the end result is
the same as a different algorithm which is also available as library. ;)

Signed-off-by: Ivo van Doorn <IvDoorn@gmail.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
drivers/net/wireless/rt2x00/Kconfig
drivers/net/wireless/rt2x00/rt2x00.h
drivers/net/wireless/rt2x00/rt2x00dev.c
drivers/net/wireless/rt2x00/rt2x00firmware.c
drivers/net/wireless/rt2x00/rt61pci.c
drivers/net/wireless/rt2x00/rt73usb.c

index da05b1faf60d359ff21dcb4a39620e83d6f4a2f7..28f3e5e94ab674f4d854894dcc13d11977c97ba7 100644 (file)
@@ -28,6 +28,7 @@ config RT2X00_LIB_USB
 config RT2X00_LIB_FIRMWARE
        boolean
        depends on RT2X00_LIB
+       select CRC_CCITT
        select CRC_ITU_T
        select FW_LOADER
 
index 7b67f9666c3bdddf83b7392cdd6ab1cc342b92f6..6a25195816db76d5eb519db42eabd0454b94c8d4 100644 (file)
@@ -551,6 +551,8 @@ enum rt2x00_flags {
         * Driver features
         */
        DRIVER_REQUIRE_FIRMWARE,
+       DRIVER_REQUIRE_FIRMWARE_CRC_ITU_T,
+       DRIVER_REQUIRE_FIRMWARE_CCITT,
        DRIVER_REQUIRE_BEACON_GUARD,
        DRIVER_REQUIRE_ATIM_QUEUE,
 
index 7620427887b6e4da78f5a4aa572cf72d92e25765..bc07c56f89b3b50e9bcacae5136166fe6b484e66 100644 (file)
@@ -1017,11 +1017,9 @@ int rt2x00lib_start(struct rt2x00_dev *rt2x00dev)
         * If this is the first interface which is added,
         * we should load the firmware now.
         */
-       if (test_bit(DRIVER_REQUIRE_FIRMWARE, &rt2x00dev->flags)) {
-               retval = rt2x00lib_load_firmware(rt2x00dev);
-               if (retval)
-                       return retval;
-       }
+       retval = rt2x00lib_load_firmware(rt2x00dev);
+       if (retval)
+               return retval;
 
        /*
         * Initialize the device.
index 442ff8047dcef7913f66c9acdf8ad5a0f8af1835..4f9fe56f4f2e42689ea1ddf07d686fcd680a4fde 100644 (file)
@@ -23,6 +23,7 @@
        Abstract: rt2x00 firmware loading routines.
  */
 
+#include <linux/crc-ccitt.h>
 #include <linux/crc-itu-t.h>
 #include <linux/kernel.h>
 #include <linux/module.h>
@@ -37,7 +38,6 @@ static int rt2x00lib_request_firmware(struct rt2x00_dev *rt2x00dev)
        char *fw_name;
        int retval;
        u16 crc;
-       u16 tmp;
 
        /*
         * Read correct firmware from harddisk.
@@ -64,17 +64,37 @@ static int rt2x00lib_request_firmware(struct rt2x00_dev *rt2x00dev)
        }
 
        /*
-        * Validate the firmware using 16 bit CRC.
-        * The last 2 bytes of the firmware are the CRC
-        * so substract those 2 bytes from the CRC checksum,
-        * and set those 2 bytes to 0 when calculating CRC.
+        * Perform crc validation on the firmware.
+        * The last 2 bytes in the firmware array are the crc checksum itself,
+        * this means that we should never pass those 2 bytes to the crc
+        * algorithm.
         */
-       tmp = 0;
-       crc = crc_itu_t(0, fw->data, fw->size - 2);
-       crc = crc_itu_t(crc, (u8 *)&tmp, 2);
+       if (test_bit(DRIVER_REQUIRE_FIRMWARE_CRC_ITU_T, &rt2x00dev->flags)) {
+               /*
+                * Use the crc itu-t algorithm.
+                * Use 0 for the last 2 bytes to complete the checksum.
+                */
+               crc = crc_itu_t(0, fw->data, fw->size - 2);
+               crc = crc_itu_t_byte(crc, 0);
+               crc = crc_itu_t_byte(crc, 0);
+       } else if (test_bit(DRIVER_REQUIRE_FIRMWARE_CCITT, &rt2x00dev->flags)) {
+               /*
+                * Use the crc ccitt algorithm.
+                * This will return the same value as the legacy driver which
+                * used bit ordering reversion on the both the firmware bytes
+                * before input input as well as on the final output.
+                * Obviously using crc ccitt directly is much more efficient.
+                */
+               crc = crc_ccitt(~0, fw->data, fw->size - 2);
+       } else {
+               ERROR(rt2x00dev, "No checksum algorithm selected "
+                     "for firmware validation.\n");
+               retval = -ENOENT;
+               goto exit;
+       }
 
        if (crc != (fw->data[fw->size - 2] << 8 | fw->data[fw->size - 1])) {
-               ERROR(rt2x00dev, "Firmware CRC error.\n");
+               ERROR(rt2x00dev, "Firmware checksum error.\n");
                retval = -ENOENT;
                goto exit;
        }
@@ -96,6 +116,9 @@ int rt2x00lib_load_firmware(struct rt2x00_dev *rt2x00dev)
 {
        int retval;
 
+       if (!test_bit(DRIVER_REQUIRE_FIRMWARE, &rt2x00dev->flags))
+               return 0;
+
        if (!rt2x00dev->fw) {
                retval = rt2x00lib_request_firmware(rt2x00dev);
                if (retval)
index 5e08541958ccddb8d86d202052a384293a4d63fe..b5ab771bbac2fe2573336e9bfd477a4e02fad91b 100644 (file)
@@ -2258,9 +2258,10 @@ static int rt61pci_probe_hw(struct rt2x00_dev *rt2x00dev)
        rt61pci_probe_hw_mode(rt2x00dev);
 
        /*
-        * This device requires firmware
+        * This device requires firmware.
         */
        __set_bit(DRIVER_REQUIRE_FIRMWARE, &rt2x00dev->flags);
+       __set_bit(DRIVER_REQUIRE_FIRMWARE_CRC_ITU_T, &rt2x00dev->flags);
 
        /*
         * Set the rssi offset.
index 4e6466b13af1c140a2602a2db4dd1ee25e867b64..8ebf3fed9815ad13080ef87681241118f637adaa 100644 (file)
@@ -1849,9 +1849,10 @@ static int rt73usb_probe_hw(struct rt2x00_dev *rt2x00dev)
        rt73usb_probe_hw_mode(rt2x00dev);
 
        /*
-        * This device requires firmware
+        * This device requires firmware.
         */
        __set_bit(DRIVER_REQUIRE_FIRMWARE, &rt2x00dev->flags);
+       __set_bit(DRIVER_REQUIRE_FIRMWARE_CRC_ITU_T, &rt2x00dev->flags);
 
        /*
         * Set the rssi offset.