ks959-sir.c 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909
  1. /*****************************************************************************
  2. *
  3. * Filename: ks959-sir.c
  4. * Version: 0.1.2
  5. * Description: Irda KingSun KS-959 USB Dongle
  6. * Status: Experimental
  7. * Author: Alex Villacís Lasso <a_villacis@palosanto.com>
  8. * with help from Domen Puncer <domen@coderock.org>
  9. *
  10. * Based on stir4200, mcs7780, kingsun-sir drivers.
  11. *
  12. * This program is free software; you can redistribute it and/or modify
  13. * it under the terms of the GNU General Public License as published by
  14. * the Free Software Foundation; either version 2 of the License.
  15. *
  16. * This program is distributed in the hope that it will be useful,
  17. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  18. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  19. * GNU General Public License for more details.
  20. *
  21. * You should have received a copy of the GNU General Public License
  22. * along with this program; if not, write to the Free Software
  23. * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  24. *
  25. *****************************************************************************/
  26. /*
  27. * Following is my most current (2007-07-17) understanding of how the Kingsun
  28. * KS-959 dongle is supposed to work. This information was deduced by
  29. * reverse-engineering and examining the USB traffic captured with USBSnoopy
  30. * from the WinXP driver. Feel free to update here as more of the dongle is
  31. * known.
  32. *
  33. * My most sincere thanks must go to Domen Puncer <domen@coderock.org> for
  34. * invaluable help in cracking the obfuscation and padding required for this
  35. * dongle.
  36. *
  37. * General: This dongle exposes one interface with one interrupt IN endpoint.
  38. * However, the interrupt endpoint is NOT used at all for this dongle. Instead,
  39. * this dongle uses control transfers for everything, including sending and
  40. * receiving the IrDA frame data. Apparently the interrupt endpoint is just a
  41. * dummy to ensure the dongle has a valid interface to present to the PC.And I
  42. * thought the DonShine dongle was weird... In addition, this dongle uses
  43. * obfuscation (?!?!), applied at the USB level, to hide the traffic, both sent
  44. * and received, from the dongle. I call it obfuscation because the XOR keying
  45. * and padding required to produce an USB traffic acceptable for the dongle can
  46. * not be explained by any other technical requirement.
  47. *
  48. * Transmission: To transmit an IrDA frame, the driver must prepare a control
  49. * URB with the following as a setup packet:
  50. * bRequestType USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE
  51. * bRequest 0x09
  52. * wValue <length of valid data before padding, little endian>
  53. * wIndex 0x0000
  54. * wLength <length of padded data>
  55. * The payload packet must be manually wrapped and escaped (as in stir4200.c),
  56. * then padded and obfuscated before being sent. Both padding and obfuscation
  57. * are implemented in the procedure obfuscate_tx_buffer(). Suffice to say, the
  58. * designer/programmer of the dongle used his name as a source for the
  59. * obfuscation. WTF?!
  60. * Apparently the dongle cannot handle payloads larger than 256 bytes. The
  61. * driver has to perform fragmentation in order to send anything larger than
  62. * this limit.
  63. *
  64. * Reception: To receive data, the driver must poll the dongle regularly (like
  65. * kingsun-sir.c) with control URBs and the following as a setup packet:
  66. * bRequestType USB_DIR_IN | USB_TYPE_CLASS | USB_RECIP_INTERFACE
  67. * bRequest 0x01
  68. * wValue 0x0200
  69. * wIndex 0x0000
  70. * wLength 0x0800 (size of available buffer)
  71. * If there is data to be read, it will be returned as the response payload.
  72. * This data is (apparently) not padded, but it is obfuscated. To de-obfuscate
  73. * it, the driver must XOR every byte, in sequence, with a value that starts at
  74. * 1 and is incremented with each byte processed, and then with 0x55. The value
  75. * incremented with each byte processed overflows as an unsigned char. The
  76. * resulting bytes form a wrapped SIR frame that is unwrapped and unescaped
  77. * as in stir4200.c The incremented value is NOT reset with each frame, but is
  78. * kept across the entire session with the dongle. Also, the dongle inserts an
  79. * extra garbage byte with value 0x95 (after decoding) every 0xff bytes, which
  80. * must be skipped.
  81. *
  82. * Speed change: To change the speed of the dongle, the driver prepares a
  83. * control URB with the following as a setup packet:
  84. * bRequestType USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE
  85. * bRequest 0x09
  86. * wValue 0x0200
  87. * wIndex 0x0001
  88. * wLength 0x0008 (length of the payload)
  89. * The payload is a 8-byte record, apparently identical to the one used in
  90. * drivers/usb/serial/cypress_m8.c to change speed:
  91. * __u32 baudSpeed;
  92. * unsigned int dataBits : 2; // 0 - 5 bits 3 - 8 bits
  93. * unsigned int : 1;
  94. * unsigned int stopBits : 1;
  95. * unsigned int parityEnable : 1;
  96. * unsigned int parityType : 1;
  97. * unsigned int : 1;
  98. * unsigned int reset : 1;
  99. * unsigned char reserved[3]; // set to 0
  100. *
  101. * For now only SIR speeds have been observed with this dongle. Therefore,
  102. * nothing is known on what changes (if any) must be done to frame wrapping /
  103. * unwrapping for higher than SIR speeds. This driver assumes no change is
  104. * necessary and announces support for all the way to 57600 bps. Although the
  105. * package announces support for up to 4MBps, tests with a Sony Ericcson K300
  106. * phone show corruption when receiving large frames at 115200 bps, the highest
  107. * speed announced by the phone. However, transmission at 115200 bps is OK. Go
  108. * figure. Since I don't know whether the phone or the dongle is at fault, max
  109. * announced speed is 57600 bps until someone produces a device that can run
  110. * at higher speeds with this dongle.
  111. */
  112. #include <linux/module.h>
  113. #include <linux/moduleparam.h>
  114. #include <linux/kernel.h>
  115. #include <linux/types.h>
  116. #include <linux/errno.h>
  117. #include <linux/init.h>
  118. #include <linux/slab.h>
  119. #include <linux/usb.h>
  120. #include <linux/device.h>
  121. #include <linux/crc32.h>
  122. #include <asm/unaligned.h>
  123. #include <asm/byteorder.h>
  124. #include <asm/uaccess.h>
  125. #include <net/irda/irda.h>
  126. #include <net/irda/wrapper.h>
  127. #include <net/irda/crc.h>
  128. #define KS959_VENDOR_ID 0x07d0
  129. #define KS959_PRODUCT_ID 0x4959
  130. /* These are the currently known USB ids */
  131. static struct usb_device_id dongles[] = {
  132. /* KingSun Co,Ltd IrDA/USB Bridge */
  133. {USB_DEVICE(KS959_VENDOR_ID, KS959_PRODUCT_ID)},
  134. {}
  135. };
  136. MODULE_DEVICE_TABLE(usb, dongles);
  137. #define KINGSUN_MTT 0x07
  138. #define KINGSUN_REQ_RECV 0x01
  139. #define KINGSUN_REQ_SEND 0x09
  140. #define KINGSUN_RCV_FIFO_SIZE 2048 /* Max length we can receive */
  141. #define KINGSUN_SND_FIFO_SIZE 2048 /* Max packet we can send */
  142. #define KINGSUN_SND_PACKET_SIZE 256 /* Max packet dongle can handle */
  143. struct ks959_speedparams {
  144. __le32 baudrate; /* baud rate, little endian */
  145. __u8 flags;
  146. __u8 reserved[3];
  147. } __packed;
  148. #define KS_DATA_5_BITS 0x00
  149. #define KS_DATA_6_BITS 0x01
  150. #define KS_DATA_7_BITS 0x02
  151. #define KS_DATA_8_BITS 0x03
  152. #define KS_STOP_BITS_1 0x00
  153. #define KS_STOP_BITS_2 0x08
  154. #define KS_PAR_DISABLE 0x00
  155. #define KS_PAR_EVEN 0x10
  156. #define KS_PAR_ODD 0x30
  157. #define KS_RESET 0x80
  158. struct ks959_cb {
  159. struct usb_device *usbdev; /* init: probe_irda */
  160. struct net_device *netdev; /* network layer */
  161. struct irlap_cb *irlap; /* The link layer we are binded to */
  162. struct qos_info qos;
  163. struct usb_ctrlrequest *tx_setuprequest;
  164. struct urb *tx_urb;
  165. __u8 *tx_buf_clear;
  166. unsigned int tx_buf_clear_used;
  167. unsigned int tx_buf_clear_sent;
  168. __u8 *tx_buf_xored;
  169. struct usb_ctrlrequest *rx_setuprequest;
  170. struct urb *rx_urb;
  171. __u8 *rx_buf;
  172. __u8 rx_variable_xormask;
  173. iobuff_t rx_unwrap_buff;
  174. struct timeval rx_time;
  175. struct usb_ctrlrequest *speed_setuprequest;
  176. struct urb *speed_urb;
  177. struct ks959_speedparams speedparams;
  178. unsigned int new_speed;
  179. spinlock_t lock;
  180. int receiving;
  181. };
  182. /* Procedure to perform the obfuscation/padding expected by the dongle
  183. *
  184. * buf_cleartext (IN) Cleartext version of the IrDA frame to transmit
  185. * len_cleartext (IN) Length of the cleartext version of IrDA frame
  186. * buf_xoredtext (OUT) Obfuscated version of frame built by proc
  187. * len_maxbuf (OUT) Maximum space available at buf_xoredtext
  188. *
  189. * (return) length of obfuscated frame with padding
  190. *
  191. * If not enough space (as indicated by len_maxbuf vs. required padding),
  192. * zero is returned
  193. *
  194. * The value of lookup_string is actually a required portion of the algorithm.
  195. * Seems the designer of the dongle wanted to state who exactly is responsible
  196. * for implementing obfuscation. Send your best (or other) wishes to him ]:-)
  197. */
  198. static unsigned int obfuscate_tx_buffer(const __u8 * buf_cleartext,
  199. unsigned int len_cleartext,
  200. __u8 * buf_xoredtext,
  201. unsigned int len_maxbuf)
  202. {
  203. unsigned int len_xoredtext;
  204. /* Calculate required length with padding, check for necessary space */
  205. len_xoredtext = ((len_cleartext + 7) & ~0x7) + 0x10;
  206. if (len_xoredtext <= len_maxbuf) {
  207. static const __u8 lookup_string[] = "wangshuofei19710";
  208. __u8 xor_mask;
  209. /* Unlike the WinXP driver, we *do* clear out the padding */
  210. memset(buf_xoredtext, 0, len_xoredtext);
  211. xor_mask = lookup_string[(len_cleartext & 0x0f) ^ 0x06] ^ 0x55;
  212. while (len_cleartext-- > 0) {
  213. *buf_xoredtext++ = *buf_cleartext++ ^ xor_mask;
  214. }
  215. } else {
  216. len_xoredtext = 0;
  217. }
  218. return len_xoredtext;
  219. }
  220. /* Callback transmission routine */
  221. static void ks959_speed_irq(struct urb *urb)
  222. {
  223. /* unlink, shutdown, unplug, other nasties */
  224. if (urb->status != 0) {
  225. err("ks959_speed_irq: urb asynchronously failed - %d",
  226. urb->status);
  227. }
  228. }
  229. /* Send a control request to change speed of the dongle */
  230. static int ks959_change_speed(struct ks959_cb *kingsun, unsigned speed)
  231. {
  232. static unsigned int supported_speeds[] = { 2400, 9600, 19200, 38400,
  233. 57600, 115200, 576000, 1152000, 4000000, 0
  234. };
  235. int err;
  236. unsigned int i;
  237. if (kingsun->speed_setuprequest == NULL || kingsun->speed_urb == NULL)
  238. return -ENOMEM;
  239. /* Check that requested speed is among the supported ones */
  240. for (i = 0; supported_speeds[i] && supported_speeds[i] != speed; i++) ;
  241. if (supported_speeds[i] == 0)
  242. return -EOPNOTSUPP;
  243. memset(&(kingsun->speedparams), 0, sizeof(struct ks959_speedparams));
  244. kingsun->speedparams.baudrate = cpu_to_le32(speed);
  245. kingsun->speedparams.flags = KS_DATA_8_BITS;
  246. /* speed_setuprequest pre-filled in ks959_probe */
  247. usb_fill_control_urb(kingsun->speed_urb, kingsun->usbdev,
  248. usb_sndctrlpipe(kingsun->usbdev, 0),
  249. (unsigned char *)kingsun->speed_setuprequest,
  250. &(kingsun->speedparams),
  251. sizeof(struct ks959_speedparams), ks959_speed_irq,
  252. kingsun);
  253. kingsun->speed_urb->status = 0;
  254. err = usb_submit_urb(kingsun->speed_urb, GFP_ATOMIC);
  255. return err;
  256. }
  257. /* Submit one fragment of an IrDA frame to the dongle */
  258. static void ks959_send_irq(struct urb *urb);
  259. static int ks959_submit_tx_fragment(struct ks959_cb *kingsun)
  260. {
  261. unsigned int padlen;
  262. unsigned int wraplen;
  263. int ret;
  264. /* Check whether current plaintext can produce a padded buffer that fits
  265. within the range handled by the dongle */
  266. wraplen = (KINGSUN_SND_PACKET_SIZE & ~0x7) - 0x10;
  267. if (wraplen > kingsun->tx_buf_clear_used)
  268. wraplen = kingsun->tx_buf_clear_used;
  269. /* Perform dongle obfuscation. Also remove the portion of the frame that
  270. was just obfuscated and will now be sent to the dongle. */
  271. padlen = obfuscate_tx_buffer(kingsun->tx_buf_clear, wraplen,
  272. kingsun->tx_buf_xored,
  273. KINGSUN_SND_PACKET_SIZE);
  274. /* Calculate how much data can be transmitted in this urb */
  275. kingsun->tx_setuprequest->wValue = cpu_to_le16(wraplen);
  276. kingsun->tx_setuprequest->wLength = cpu_to_le16(padlen);
  277. /* Rest of the fields were filled in ks959_probe */
  278. usb_fill_control_urb(kingsun->tx_urb, kingsun->usbdev,
  279. usb_sndctrlpipe(kingsun->usbdev, 0),
  280. (unsigned char *)kingsun->tx_setuprequest,
  281. kingsun->tx_buf_xored, padlen,
  282. ks959_send_irq, kingsun);
  283. kingsun->tx_urb->status = 0;
  284. ret = usb_submit_urb(kingsun->tx_urb, GFP_ATOMIC);
  285. /* Remember how much data was sent, in order to update at callback */
  286. kingsun->tx_buf_clear_sent = (ret == 0) ? wraplen : 0;
  287. return ret;
  288. }
  289. /* Callback transmission routine */
  290. static void ks959_send_irq(struct urb *urb)
  291. {
  292. struct ks959_cb *kingsun = urb->context;
  293. struct net_device *netdev = kingsun->netdev;
  294. int ret = 0;
  295. /* in process of stopping, just drop data */
  296. if (!netif_running(kingsun->netdev)) {
  297. err("ks959_send_irq: Network not running!");
  298. return;
  299. }
  300. /* unlink, shutdown, unplug, other nasties */
  301. if (urb->status != 0) {
  302. err("ks959_send_irq: urb asynchronously failed - %d",
  303. urb->status);
  304. return;
  305. }
  306. if (kingsun->tx_buf_clear_used > 0) {
  307. /* Update data remaining to be sent */
  308. if (kingsun->tx_buf_clear_sent < kingsun->tx_buf_clear_used) {
  309. memmove(kingsun->tx_buf_clear,
  310. kingsun->tx_buf_clear +
  311. kingsun->tx_buf_clear_sent,
  312. kingsun->tx_buf_clear_used -
  313. kingsun->tx_buf_clear_sent);
  314. }
  315. kingsun->tx_buf_clear_used -= kingsun->tx_buf_clear_sent;
  316. kingsun->tx_buf_clear_sent = 0;
  317. if (kingsun->tx_buf_clear_used > 0) {
  318. /* There is more data to be sent */
  319. if ((ret = ks959_submit_tx_fragment(kingsun)) != 0) {
  320. err("ks959_send_irq: failed tx_urb submit: %d",
  321. ret);
  322. switch (ret) {
  323. case -ENODEV:
  324. case -EPIPE:
  325. break;
  326. default:
  327. netdev->stats.tx_errors++;
  328. netif_start_queue(netdev);
  329. }
  330. }
  331. } else {
  332. /* All data sent, send next speed && wake network queue */
  333. if (kingsun->new_speed != -1 &&
  334. cpu_to_le32(kingsun->new_speed) !=
  335. kingsun->speedparams.baudrate)
  336. ks959_change_speed(kingsun, kingsun->new_speed);
  337. netif_wake_queue(netdev);
  338. }
  339. }
  340. }
  341. /*
  342. * Called from net/core when new frame is available.
  343. */
  344. static netdev_tx_t ks959_hard_xmit(struct sk_buff *skb,
  345. struct net_device *netdev)
  346. {
  347. struct ks959_cb *kingsun;
  348. unsigned int wraplen;
  349. int ret = 0;
  350. netif_stop_queue(netdev);
  351. /* the IRDA wrapping routines don't deal with non linear skb */
  352. SKB_LINEAR_ASSERT(skb);
  353. kingsun = netdev_priv(netdev);
  354. spin_lock(&kingsun->lock);
  355. kingsun->new_speed = irda_get_next_speed(skb);
  356. /* Append data to the end of whatever data remains to be transmitted */
  357. wraplen =
  358. async_wrap_skb(skb, kingsun->tx_buf_clear, KINGSUN_SND_FIFO_SIZE);
  359. kingsun->tx_buf_clear_used = wraplen;
  360. if ((ret = ks959_submit_tx_fragment(kingsun)) != 0) {
  361. err("ks959_hard_xmit: failed tx_urb submit: %d", ret);
  362. switch (ret) {
  363. case -ENODEV:
  364. case -EPIPE:
  365. break;
  366. default:
  367. netdev->stats.tx_errors++;
  368. netif_start_queue(netdev);
  369. }
  370. } else {
  371. netdev->stats.tx_packets++;
  372. netdev->stats.tx_bytes += skb->len;
  373. }
  374. dev_kfree_skb(skb);
  375. spin_unlock(&kingsun->lock);
  376. return NETDEV_TX_OK;
  377. }
  378. /* Receive callback function */
  379. static void ks959_rcv_irq(struct urb *urb)
  380. {
  381. struct ks959_cb *kingsun = urb->context;
  382. int ret;
  383. /* in process of stopping, just drop data */
  384. if (!netif_running(kingsun->netdev)) {
  385. kingsun->receiving = 0;
  386. return;
  387. }
  388. /* unlink, shutdown, unplug, other nasties */
  389. if (urb->status != 0) {
  390. err("kingsun_rcv_irq: urb asynchronously failed - %d",
  391. urb->status);
  392. kingsun->receiving = 0;
  393. return;
  394. }
  395. if (urb->actual_length > 0) {
  396. __u8 *bytes = urb->transfer_buffer;
  397. unsigned int i;
  398. for (i = 0; i < urb->actual_length; i++) {
  399. /* De-obfuscation implemented here: variable portion of
  400. xormask is incremented, and then used with the encoded
  401. byte for the XOR. The result of the operation is used
  402. to unwrap the SIR frame. */
  403. kingsun->rx_variable_xormask++;
  404. bytes[i] =
  405. bytes[i] ^ kingsun->rx_variable_xormask ^ 0x55u;
  406. /* rx_variable_xormask doubles as an index counter so we
  407. can skip the byte at 0xff (wrapped around to 0).
  408. */
  409. if (kingsun->rx_variable_xormask != 0) {
  410. async_unwrap_char(kingsun->netdev,
  411. &kingsun->netdev->stats,
  412. &kingsun->rx_unwrap_buff,
  413. bytes[i]);
  414. }
  415. }
  416. do_gettimeofday(&kingsun->rx_time);
  417. kingsun->receiving =
  418. (kingsun->rx_unwrap_buff.state != OUTSIDE_FRAME) ? 1 : 0;
  419. }
  420. /* This urb has already been filled in kingsun_net_open. Setup
  421. packet must be re-filled, but it is assumed that urb keeps the
  422. pointer to the initial setup packet, as well as the payload buffer.
  423. Setup packet is already pre-filled at ks959_probe.
  424. */
  425. urb->status = 0;
  426. ret = usb_submit_urb(urb, GFP_ATOMIC);
  427. }
  428. /*
  429. * Function kingsun_net_open (dev)
  430. *
  431. * Network device is taken up. Usually this is done by "ifconfig irda0 up"
  432. */
  433. static int ks959_net_open(struct net_device *netdev)
  434. {
  435. struct ks959_cb *kingsun = netdev_priv(netdev);
  436. int err = -ENOMEM;
  437. char hwname[16];
  438. /* At this point, urbs are NULL, and skb is NULL (see kingsun_probe) */
  439. kingsun->receiving = 0;
  440. /* Initialize for SIR to copy data directly into skb. */
  441. kingsun->rx_unwrap_buff.in_frame = FALSE;
  442. kingsun->rx_unwrap_buff.state = OUTSIDE_FRAME;
  443. kingsun->rx_unwrap_buff.truesize = IRDA_SKB_MAX_MTU;
  444. kingsun->rx_unwrap_buff.skb = dev_alloc_skb(IRDA_SKB_MAX_MTU);
  445. if (!kingsun->rx_unwrap_buff.skb)
  446. goto free_mem;
  447. skb_reserve(kingsun->rx_unwrap_buff.skb, 1);
  448. kingsun->rx_unwrap_buff.head = kingsun->rx_unwrap_buff.skb->data;
  449. do_gettimeofday(&kingsun->rx_time);
  450. kingsun->rx_urb = usb_alloc_urb(0, GFP_KERNEL);
  451. if (!kingsun->rx_urb)
  452. goto free_mem;
  453. kingsun->tx_urb = usb_alloc_urb(0, GFP_KERNEL);
  454. if (!kingsun->tx_urb)
  455. goto free_mem;
  456. kingsun->speed_urb = usb_alloc_urb(0, GFP_KERNEL);
  457. if (!kingsun->speed_urb)
  458. goto free_mem;
  459. /* Initialize speed for dongle */
  460. kingsun->new_speed = 9600;
  461. err = ks959_change_speed(kingsun, 9600);
  462. if (err < 0)
  463. goto free_mem;
  464. /*
  465. * Now that everything should be initialized properly,
  466. * Open new IrLAP layer instance to take care of us...
  467. */
  468. sprintf(hwname, "usb#%d", kingsun->usbdev->devnum);
  469. kingsun->irlap = irlap_open(netdev, &kingsun->qos, hwname);
  470. if (!kingsun->irlap) {
  471. err("ks959-sir: irlap_open failed");
  472. goto free_mem;
  473. }
  474. /* Start reception. Setup request already pre-filled in ks959_probe */
  475. usb_fill_control_urb(kingsun->rx_urb, kingsun->usbdev,
  476. usb_rcvctrlpipe(kingsun->usbdev, 0),
  477. (unsigned char *)kingsun->rx_setuprequest,
  478. kingsun->rx_buf, KINGSUN_RCV_FIFO_SIZE,
  479. ks959_rcv_irq, kingsun);
  480. kingsun->rx_urb->status = 0;
  481. err = usb_submit_urb(kingsun->rx_urb, GFP_KERNEL);
  482. if (err) {
  483. err("ks959-sir: first urb-submit failed: %d", err);
  484. goto close_irlap;
  485. }
  486. netif_start_queue(netdev);
  487. /* Situation at this point:
  488. - all work buffers allocated
  489. - urbs allocated and ready to fill
  490. - max rx packet known (in max_rx)
  491. - unwrap state machine initialized, in state outside of any frame
  492. - receive request in progress
  493. - IrLAP layer started, about to hand over packets to send
  494. */
  495. return 0;
  496. close_irlap:
  497. irlap_close(kingsun->irlap);
  498. free_mem:
  499. usb_free_urb(kingsun->speed_urb);
  500. kingsun->speed_urb = NULL;
  501. usb_free_urb(kingsun->tx_urb);
  502. kingsun->tx_urb = NULL;
  503. usb_free_urb(kingsun->rx_urb);
  504. kingsun->rx_urb = NULL;
  505. if (kingsun->rx_unwrap_buff.skb) {
  506. kfree_skb(kingsun->rx_unwrap_buff.skb);
  507. kingsun->rx_unwrap_buff.skb = NULL;
  508. kingsun->rx_unwrap_buff.head = NULL;
  509. }
  510. return err;
  511. }
  512. /*
  513. * Function kingsun_net_close (kingsun)
  514. *
  515. * Network device is taken down. Usually this is done by
  516. * "ifconfig irda0 down"
  517. */
  518. static int ks959_net_close(struct net_device *netdev)
  519. {
  520. struct ks959_cb *kingsun = netdev_priv(netdev);
  521. /* Stop transmit processing */
  522. netif_stop_queue(netdev);
  523. /* Mop up receive && transmit urb's */
  524. usb_kill_urb(kingsun->tx_urb);
  525. usb_free_urb(kingsun->tx_urb);
  526. kingsun->tx_urb = NULL;
  527. usb_kill_urb(kingsun->speed_urb);
  528. usb_free_urb(kingsun->speed_urb);
  529. kingsun->speed_urb = NULL;
  530. usb_kill_urb(kingsun->rx_urb);
  531. usb_free_urb(kingsun->rx_urb);
  532. kingsun->rx_urb = NULL;
  533. kfree_skb(kingsun->rx_unwrap_buff.skb);
  534. kingsun->rx_unwrap_buff.skb = NULL;
  535. kingsun->rx_unwrap_buff.head = NULL;
  536. kingsun->rx_unwrap_buff.in_frame = FALSE;
  537. kingsun->rx_unwrap_buff.state = OUTSIDE_FRAME;
  538. kingsun->receiving = 0;
  539. /* Stop and remove instance of IrLAP */
  540. if (kingsun->irlap)
  541. irlap_close(kingsun->irlap);
  542. kingsun->irlap = NULL;
  543. return 0;
  544. }
  545. /*
  546. * IOCTLs : Extra out-of-band network commands...
  547. */
  548. static int ks959_net_ioctl(struct net_device *netdev, struct ifreq *rq, int cmd)
  549. {
  550. struct if_irda_req *irq = (struct if_irda_req *)rq;
  551. struct ks959_cb *kingsun = netdev_priv(netdev);
  552. int ret = 0;
  553. switch (cmd) {
  554. case SIOCSBANDWIDTH: /* Set bandwidth */
  555. if (!capable(CAP_NET_ADMIN))
  556. return -EPERM;
  557. /* Check if the device is still there */
  558. if (netif_device_present(kingsun->netdev))
  559. return ks959_change_speed(kingsun, irq->ifr_baudrate);
  560. break;
  561. case SIOCSMEDIABUSY: /* Set media busy */
  562. if (!capable(CAP_NET_ADMIN))
  563. return -EPERM;
  564. /* Check if the IrDA stack is still there */
  565. if (netif_running(kingsun->netdev))
  566. irda_device_set_media_busy(kingsun->netdev, TRUE);
  567. break;
  568. case SIOCGRECEIVING:
  569. /* Only approximately true */
  570. irq->ifr_receiving = kingsun->receiving;
  571. break;
  572. default:
  573. ret = -EOPNOTSUPP;
  574. }
  575. return ret;
  576. }
  577. static const struct net_device_ops ks959_ops = {
  578. .ndo_start_xmit = ks959_hard_xmit,
  579. .ndo_open = ks959_net_open,
  580. .ndo_stop = ks959_net_close,
  581. .ndo_do_ioctl = ks959_net_ioctl,
  582. };
  583. /*
  584. * This routine is called by the USB subsystem for each new device
  585. * in the system. We need to check if the device is ours, and in
  586. * this case start handling it.
  587. */
  588. static int ks959_probe(struct usb_interface *intf,
  589. const struct usb_device_id *id)
  590. {
  591. struct usb_device *dev = interface_to_usbdev(intf);
  592. struct ks959_cb *kingsun = NULL;
  593. struct net_device *net = NULL;
  594. int ret = -ENOMEM;
  595. /* Allocate network device container. */
  596. net = alloc_irdadev(sizeof(*kingsun));
  597. if (!net)
  598. goto err_out1;
  599. SET_NETDEV_DEV(net, &intf->dev);
  600. kingsun = netdev_priv(net);
  601. kingsun->netdev = net;
  602. kingsun->usbdev = dev;
  603. kingsun->irlap = NULL;
  604. kingsun->tx_setuprequest = NULL;
  605. kingsun->tx_urb = NULL;
  606. kingsun->tx_buf_clear = NULL;
  607. kingsun->tx_buf_xored = NULL;
  608. kingsun->tx_buf_clear_used = 0;
  609. kingsun->tx_buf_clear_sent = 0;
  610. kingsun->rx_setuprequest = NULL;
  611. kingsun->rx_urb = NULL;
  612. kingsun->rx_buf = NULL;
  613. kingsun->rx_variable_xormask = 0;
  614. kingsun->rx_unwrap_buff.in_frame = FALSE;
  615. kingsun->rx_unwrap_buff.state = OUTSIDE_FRAME;
  616. kingsun->rx_unwrap_buff.skb = NULL;
  617. kingsun->receiving = 0;
  618. spin_lock_init(&kingsun->lock);
  619. kingsun->speed_setuprequest = NULL;
  620. kingsun->speed_urb = NULL;
  621. kingsun->speedparams.baudrate = 0;
  622. /* Allocate input buffer */
  623. kingsun->rx_buf = kmalloc(KINGSUN_RCV_FIFO_SIZE, GFP_KERNEL);
  624. if (!kingsun->rx_buf)
  625. goto free_mem;
  626. /* Allocate input setup packet */
  627. kingsun->rx_setuprequest =
  628. kmalloc(sizeof(struct usb_ctrlrequest), GFP_KERNEL);
  629. if (!kingsun->rx_setuprequest)
  630. goto free_mem;
  631. kingsun->rx_setuprequest->bRequestType =
  632. USB_DIR_IN | USB_TYPE_CLASS | USB_RECIP_INTERFACE;
  633. kingsun->rx_setuprequest->bRequest = KINGSUN_REQ_RECV;
  634. kingsun->rx_setuprequest->wValue = cpu_to_le16(0x0200);
  635. kingsun->rx_setuprequest->wIndex = 0;
  636. kingsun->rx_setuprequest->wLength = cpu_to_le16(KINGSUN_RCV_FIFO_SIZE);
  637. /* Allocate output buffer */
  638. kingsun->tx_buf_clear = kmalloc(KINGSUN_SND_FIFO_SIZE, GFP_KERNEL);
  639. if (!kingsun->tx_buf_clear)
  640. goto free_mem;
  641. kingsun->tx_buf_xored = kmalloc(KINGSUN_SND_PACKET_SIZE, GFP_KERNEL);
  642. if (!kingsun->tx_buf_xored)
  643. goto free_mem;
  644. /* Allocate and initialize output setup packet */
  645. kingsun->tx_setuprequest =
  646. kmalloc(sizeof(struct usb_ctrlrequest), GFP_KERNEL);
  647. if (!kingsun->tx_setuprequest)
  648. goto free_mem;
  649. kingsun->tx_setuprequest->bRequestType =
  650. USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE;
  651. kingsun->tx_setuprequest->bRequest = KINGSUN_REQ_SEND;
  652. kingsun->tx_setuprequest->wValue = 0;
  653. kingsun->tx_setuprequest->wIndex = 0;
  654. kingsun->tx_setuprequest->wLength = 0;
  655. /* Allocate and initialize speed setup packet */
  656. kingsun->speed_setuprequest =
  657. kmalloc(sizeof(struct usb_ctrlrequest), GFP_KERNEL);
  658. if (!kingsun->speed_setuprequest)
  659. goto free_mem;
  660. kingsun->speed_setuprequest->bRequestType =
  661. USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE;
  662. kingsun->speed_setuprequest->bRequest = KINGSUN_REQ_SEND;
  663. kingsun->speed_setuprequest->wValue = cpu_to_le16(0x0200);
  664. kingsun->speed_setuprequest->wIndex = cpu_to_le16(0x0001);
  665. kingsun->speed_setuprequest->wLength =
  666. cpu_to_le16(sizeof(struct ks959_speedparams));
  667. printk(KERN_INFO "KingSun KS-959 IRDA/USB found at address %d, "
  668. "Vendor: %x, Product: %x\n",
  669. dev->devnum, le16_to_cpu(dev->descriptor.idVendor),
  670. le16_to_cpu(dev->descriptor.idProduct));
  671. /* Initialize QoS for this device */
  672. irda_init_max_qos_capabilies(&kingsun->qos);
  673. /* Baud rates known to be supported. Please uncomment if devices (other
  674. than a SonyEriccson K300 phone) can be shown to support higher speed
  675. with this dongle.
  676. */
  677. kingsun->qos.baud_rate.bits =
  678. IR_2400 | IR_9600 | IR_19200 | IR_38400 | IR_57600;
  679. kingsun->qos.min_turn_time.bits &= KINGSUN_MTT;
  680. irda_qos_bits_to_value(&kingsun->qos);
  681. /* Override the network functions we need to use */
  682. net->netdev_ops = &ks959_ops;
  683. ret = register_netdev(net);
  684. if (ret != 0)
  685. goto free_mem;
  686. dev_info(&net->dev, "IrDA: Registered KingSun KS-959 device %s\n",
  687. net->name);
  688. usb_set_intfdata(intf, kingsun);
  689. /* Situation at this point:
  690. - all work buffers allocated
  691. - setup requests pre-filled
  692. - urbs not allocated, set to NULL
  693. - max rx packet known (is KINGSUN_FIFO_SIZE)
  694. - unwrap state machine (partially) initialized, but skb == NULL
  695. */
  696. return 0;
  697. free_mem:
  698. kfree(kingsun->speed_setuprequest);
  699. kfree(kingsun->tx_setuprequest);
  700. kfree(kingsun->tx_buf_xored);
  701. kfree(kingsun->tx_buf_clear);
  702. kfree(kingsun->rx_setuprequest);
  703. kfree(kingsun->rx_buf);
  704. free_netdev(net);
  705. err_out1:
  706. return ret;
  707. }
  708. /*
  709. * The current device is removed, the USB layer tell us to shut it down...
  710. */
  711. static void ks959_disconnect(struct usb_interface *intf)
  712. {
  713. struct ks959_cb *kingsun = usb_get_intfdata(intf);
  714. if (!kingsun)
  715. return;
  716. unregister_netdev(kingsun->netdev);
  717. /* Mop up receive && transmit urb's */
  718. if (kingsun->speed_urb != NULL) {
  719. usb_kill_urb(kingsun->speed_urb);
  720. usb_free_urb(kingsun->speed_urb);
  721. kingsun->speed_urb = NULL;
  722. }
  723. if (kingsun->tx_urb != NULL) {
  724. usb_kill_urb(kingsun->tx_urb);
  725. usb_free_urb(kingsun->tx_urb);
  726. kingsun->tx_urb = NULL;
  727. }
  728. if (kingsun->rx_urb != NULL) {
  729. usb_kill_urb(kingsun->rx_urb);
  730. usb_free_urb(kingsun->rx_urb);
  731. kingsun->rx_urb = NULL;
  732. }
  733. kfree(kingsun->speed_setuprequest);
  734. kfree(kingsun->tx_setuprequest);
  735. kfree(kingsun->tx_buf_xored);
  736. kfree(kingsun->tx_buf_clear);
  737. kfree(kingsun->rx_setuprequest);
  738. kfree(kingsun->rx_buf);
  739. free_netdev(kingsun->netdev);
  740. usb_set_intfdata(intf, NULL);
  741. }
  742. #ifdef CONFIG_PM
  743. /* USB suspend, so power off the transmitter/receiver */
  744. static int ks959_suspend(struct usb_interface *intf, pm_message_t message)
  745. {
  746. struct ks959_cb *kingsun = usb_get_intfdata(intf);
  747. netif_device_detach(kingsun->netdev);
  748. if (kingsun->speed_urb != NULL)
  749. usb_kill_urb(kingsun->speed_urb);
  750. if (kingsun->tx_urb != NULL)
  751. usb_kill_urb(kingsun->tx_urb);
  752. if (kingsun->rx_urb != NULL)
  753. usb_kill_urb(kingsun->rx_urb);
  754. return 0;
  755. }
  756. /* Coming out of suspend, so reset hardware */
  757. static int ks959_resume(struct usb_interface *intf)
  758. {
  759. struct ks959_cb *kingsun = usb_get_intfdata(intf);
  760. if (kingsun->rx_urb != NULL) {
  761. /* Setup request already filled in ks959_probe */
  762. usb_submit_urb(kingsun->rx_urb, GFP_KERNEL);
  763. }
  764. netif_device_attach(kingsun->netdev);
  765. return 0;
  766. }
  767. #endif
  768. /*
  769. * USB device callbacks
  770. */
  771. static struct usb_driver irda_driver = {
  772. .name = "ks959-sir",
  773. .probe = ks959_probe,
  774. .disconnect = ks959_disconnect,
  775. .id_table = dongles,
  776. #ifdef CONFIG_PM
  777. .suspend = ks959_suspend,
  778. .resume = ks959_resume,
  779. #endif
  780. };
  781. module_usb_driver(irda_driver);
  782. MODULE_AUTHOR("Alex Villacís Lasso <a_villacis@palosanto.com>");
  783. MODULE_DESCRIPTION("IrDA-USB Dongle Driver for KingSun KS-959");
  784. MODULE_LICENSE("GPL");