cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

ir_toy.c (13269B)


      1// SPDX-License-Identifier: GPL-2.0-or-later
      2
      3/*
      4 * Infrared Toy and IR Droid RC core driver
      5 *
      6 * Copyright (C) 2020 Sean Young <sean@mess.org>
      7 *
      8 * http://dangerousprototypes.com/docs/USB_IR_Toy:_Sampling_mode
      9 *
     10 * This driver is based on the lirc driver which can be found here:
     11 * https://sourceforge.net/p/lirc/git/ci/master/tree/plugins/irtoy.c
     12 * Copyright (C) 2011 Peter Kooiman <pkooiman@gmail.com>
     13 */
     14
     15#include <asm/unaligned.h>
     16#include <linux/completion.h>
     17#include <linux/kernel.h>
     18#include <linux/module.h>
     19#include <linux/usb.h>
     20#include <linux/slab.h>
     21#include <linux/usb/input.h>
     22
     23#include <media/rc-core.h>
     24
     25static const u8 COMMAND_VERSION[] = { 'v' };
     26// End transmit and repeat reset command so we exit sump mode
     27static const u8 COMMAND_RESET[] = { 0xff, 0xff, 0, 0, 0, 0, 0 };
     28static const u8 COMMAND_SMODE_ENTER[] = { 's' };
     29static const u8 COMMAND_SMODE_EXIT[] = { 0 };
     30static const u8 COMMAND_TXSTART[] = { 0x26, 0x24, 0x25, 0x03 };
     31
     32#define REPLY_XMITCOUNT 't'
     33#define REPLY_XMITSUCCESS 'C'
     34#define REPLY_VERSION 'V'
     35#define REPLY_SAMPLEMODEPROTO 'S'
     36
     37#define TIMEOUT 500
     38
     39#define LEN_XMITRES 3
     40#define LEN_VERSION 4
     41#define LEN_SAMPLEMODEPROTO 3
     42
     43#define MIN_FW_VERSION 20
     44#define UNIT_US 21
     45#define MAX_TIMEOUT_US (UNIT_US * U16_MAX)
     46
     47#define MAX_PACKET 64
     48
     49enum state {
     50	STATE_IRDATA,
     51	STATE_COMMAND_NO_RESP,
     52	STATE_COMMAND,
     53	STATE_TX,
     54};
     55
     56struct irtoy {
     57	struct device *dev;
     58	struct usb_device *usbdev;
     59
     60	struct rc_dev *rc;
     61	struct urb *urb_in, *urb_out;
     62
     63	u8 *in;
     64	u8 *out;
     65	struct completion command_done;
     66
     67	bool pulse;
     68	enum state state;
     69
     70	void *tx_buf;
     71	uint tx_len;
     72
     73	uint emitted;
     74	uint hw_version;
     75	uint sw_version;
     76	uint proto_version;
     77
     78	char phys[64];
     79};
     80
     81static void irtoy_response(struct irtoy *irtoy, u32 len)
     82{
     83	switch (irtoy->state) {
     84	case STATE_COMMAND:
     85		if (len == LEN_VERSION && irtoy->in[0] == REPLY_VERSION) {
     86			uint version;
     87
     88			irtoy->in[LEN_VERSION] = 0;
     89
     90			if (kstrtouint(irtoy->in + 1, 10, &version)) {
     91				dev_err(irtoy->dev, "invalid version %*phN. Please make sure you are using firmware v20 or higher",
     92					LEN_VERSION, irtoy->in);
     93				break;
     94			}
     95
     96			dev_dbg(irtoy->dev, "version %s\n", irtoy->in);
     97
     98			irtoy->hw_version = version / 100;
     99			irtoy->sw_version = version % 100;
    100
    101			irtoy->state = STATE_IRDATA;
    102			complete(&irtoy->command_done);
    103		} else if (len == LEN_SAMPLEMODEPROTO &&
    104			   irtoy->in[0] == REPLY_SAMPLEMODEPROTO) {
    105			uint version;
    106
    107			irtoy->in[LEN_SAMPLEMODEPROTO] = 0;
    108
    109			if (kstrtouint(irtoy->in + 1, 10, &version)) {
    110				dev_err(irtoy->dev, "invalid sample mode response %*phN",
    111					LEN_SAMPLEMODEPROTO, irtoy->in);
    112				return;
    113			}
    114
    115			dev_dbg(irtoy->dev, "protocol %s\n", irtoy->in);
    116
    117			irtoy->proto_version = version;
    118
    119			irtoy->state = STATE_IRDATA;
    120			complete(&irtoy->command_done);
    121		} else {
    122			dev_err(irtoy->dev, "unexpected response to command: %*phN\n",
    123				len, irtoy->in);
    124		}
    125		break;
    126	case STATE_COMMAND_NO_RESP:
    127	case STATE_IRDATA: {
    128		struct ir_raw_event rawir = { .pulse = irtoy->pulse };
    129		__be16 *in = (__be16 *)irtoy->in;
    130		int i;
    131
    132		for (i = 0; i < len / sizeof(__be16); i++) {
    133			u16 v = be16_to_cpu(in[i]);
    134
    135			if (v == 0xffff) {
    136				rawir.pulse = false;
    137			} else {
    138				rawir.duration = v * UNIT_US;
    139				ir_raw_event_store_with_timeout(irtoy->rc,
    140								&rawir);
    141			}
    142
    143			rawir.pulse = !rawir.pulse;
    144		}
    145
    146		irtoy->pulse = rawir.pulse;
    147
    148		ir_raw_event_handle(irtoy->rc);
    149		break;
    150	}
    151	case STATE_TX:
    152		if (irtoy->tx_len == 0) {
    153			if (len == LEN_XMITRES &&
    154			    irtoy->in[0] == REPLY_XMITCOUNT) {
    155				u16 emitted = get_unaligned_be16(irtoy->in + 1);
    156
    157				dev_dbg(irtoy->dev, "emitted:%u\n", emitted);
    158
    159				irtoy->emitted = emitted;
    160			} else if (len == 1 &&
    161				   irtoy->in[0] == REPLY_XMITSUCCESS) {
    162				irtoy->state = STATE_IRDATA;
    163				complete(&irtoy->command_done);
    164			}
    165		} else {
    166			// send next part of tx buffer
    167			uint space = irtoy->in[0];
    168			uint buf_len;
    169			int err;
    170
    171			if (len != 1 || space > MAX_PACKET || space == 0) {
    172				dev_dbg(irtoy->dev, "packet length expected: %*phN\n",
    173					len, irtoy->in);
    174				break;
    175			}
    176
    177			buf_len = min(space, irtoy->tx_len);
    178
    179			dev_dbg(irtoy->dev, "remaining:%u sending:%u\n",
    180				irtoy->tx_len, buf_len);
    181
    182			memcpy(irtoy->out, irtoy->tx_buf, buf_len);
    183			irtoy->urb_out->transfer_buffer_length = buf_len;
    184			err = usb_submit_urb(irtoy->urb_out, GFP_ATOMIC);
    185			if (err != 0) {
    186				dev_err(irtoy->dev, "fail to submit tx buf urb: %d\n",
    187					err);
    188				irtoy->state = STATE_IRDATA;
    189				complete(&irtoy->command_done);
    190				break;
    191			}
    192
    193			irtoy->tx_buf += buf_len;
    194			irtoy->tx_len -= buf_len;
    195		}
    196		break;
    197	}
    198}
    199
    200static void irtoy_out_callback(struct urb *urb)
    201{
    202	struct irtoy *irtoy = urb->context;
    203
    204	if (urb->status == 0) {
    205		if (irtoy->state == STATE_COMMAND_NO_RESP)
    206			complete(&irtoy->command_done);
    207	} else {
    208		dev_warn(irtoy->dev, "out urb status: %d\n", urb->status);
    209	}
    210}
    211
    212static void irtoy_in_callback(struct urb *urb)
    213{
    214	struct irtoy *irtoy = urb->context;
    215	int ret;
    216
    217	switch (urb->status) {
    218	case 0:
    219		irtoy_response(irtoy, urb->actual_length);
    220		break;
    221	case -ECONNRESET:
    222	case -ENOENT:
    223	case -ESHUTDOWN:
    224	case -EPROTO:
    225	case -EPIPE:
    226		usb_unlink_urb(urb);
    227		return;
    228	default:
    229		dev_dbg(irtoy->dev, "in urb status: %d\n", urb->status);
    230	}
    231
    232	ret = usb_submit_urb(urb, GFP_ATOMIC);
    233	if (ret && ret != -ENODEV)
    234		dev_warn(irtoy->dev, "failed to resubmit urb: %d\n", ret);
    235}
    236
    237static int irtoy_command(struct irtoy *irtoy, const u8 *cmd, int cmd_len,
    238			 enum state state)
    239{
    240	int err;
    241
    242	init_completion(&irtoy->command_done);
    243
    244	irtoy->state = state;
    245
    246	memcpy(irtoy->out, cmd, cmd_len);
    247	irtoy->urb_out->transfer_buffer_length = cmd_len;
    248
    249	err = usb_submit_urb(irtoy->urb_out, GFP_KERNEL);
    250	if (err != 0)
    251		return err;
    252
    253	if (!wait_for_completion_timeout(&irtoy->command_done,
    254					 msecs_to_jiffies(TIMEOUT))) {
    255		usb_kill_urb(irtoy->urb_out);
    256		return -ETIMEDOUT;
    257	}
    258
    259	return 0;
    260}
    261
    262static int irtoy_setup(struct irtoy *irtoy)
    263{
    264	int err;
    265
    266	err = irtoy_command(irtoy, COMMAND_RESET, sizeof(COMMAND_RESET),
    267			    STATE_COMMAND_NO_RESP);
    268	if (err != 0) {
    269		dev_err(irtoy->dev, "could not write reset command: %d\n",
    270			err);
    271		return err;
    272	}
    273
    274	usleep_range(50, 50);
    275
    276	// get version
    277	err = irtoy_command(irtoy, COMMAND_VERSION, sizeof(COMMAND_VERSION),
    278			    STATE_COMMAND);
    279	if (err) {
    280		dev_err(irtoy->dev, "could not write version command: %d\n",
    281			err);
    282		return err;
    283	}
    284
    285	// enter sample mode
    286	err = irtoy_command(irtoy, COMMAND_SMODE_ENTER,
    287			    sizeof(COMMAND_SMODE_ENTER), STATE_COMMAND);
    288	if (err)
    289		dev_err(irtoy->dev, "could not write sample command: %d\n",
    290			err);
    291
    292	return err;
    293}
    294
    295/*
    296 * When sending IR, it is imperative that we send the IR data as quickly
    297 * as possible to the device, so it does not run out of IR data and
    298 * introduce gaps. Allocate the buffer here, and then feed the data from
    299 * the urb callback handler.
    300 */
    301static int irtoy_tx(struct rc_dev *rc, uint *txbuf, uint count)
    302{
    303	struct irtoy *irtoy = rc->priv;
    304	unsigned int i, size;
    305	__be16 *buf;
    306	int err;
    307
    308	size = sizeof(u16) * (count + 1);
    309	buf = kmalloc(size, GFP_KERNEL);
    310	if (!buf)
    311		return -ENOMEM;
    312
    313	for (i = 0; i < count; i++) {
    314		u16 v = DIV_ROUND_CLOSEST(txbuf[i], UNIT_US);
    315
    316		if (!v)
    317			v = 1;
    318		buf[i] = cpu_to_be16(v);
    319	}
    320
    321	buf[count] = cpu_to_be16(0xffff);
    322
    323	irtoy->tx_buf = buf;
    324	irtoy->tx_len = size;
    325	irtoy->emitted = 0;
    326
    327	// There is an issue where if the unit is receiving IR while the
    328	// first TXSTART command is sent, the device might end up hanging
    329	// with its led on. It does not respond to any command when this
    330	// happens. To work around this, re-enter sample mode.
    331	err = irtoy_command(irtoy, COMMAND_SMODE_EXIT,
    332			    sizeof(COMMAND_SMODE_EXIT), STATE_COMMAND_NO_RESP);
    333	if (err) {
    334		dev_err(irtoy->dev, "exit sample mode: %d\n", err);
    335		return err;
    336	}
    337
    338	err = irtoy_command(irtoy, COMMAND_SMODE_ENTER,
    339			    sizeof(COMMAND_SMODE_ENTER), STATE_COMMAND);
    340	if (err) {
    341		dev_err(irtoy->dev, "enter sample mode: %d\n", err);
    342		return err;
    343	}
    344
    345	err = irtoy_command(irtoy, COMMAND_TXSTART, sizeof(COMMAND_TXSTART),
    346			    STATE_TX);
    347	kfree(buf);
    348
    349	if (err) {
    350		dev_err(irtoy->dev, "failed to send tx start command: %d\n",
    351			err);
    352		// not sure what state the device is in, reset it
    353		irtoy_setup(irtoy);
    354		return err;
    355	}
    356
    357	if (size != irtoy->emitted) {
    358		dev_err(irtoy->dev, "expected %u emitted, got %u\n", size,
    359			irtoy->emitted);
    360		// not sure what state the device is in, reset it
    361		irtoy_setup(irtoy);
    362		return -EINVAL;
    363	}
    364
    365	return count;
    366}
    367
    368static int irtoy_tx_carrier(struct rc_dev *rc, uint32_t carrier)
    369{
    370	struct irtoy *irtoy = rc->priv;
    371	u8 buf[3];
    372	int err;
    373
    374	if (carrier < 11800)
    375		return -EINVAL;
    376
    377	buf[0] = 0x06;
    378	buf[1] = DIV_ROUND_CLOSEST(48000000, 16 * carrier) - 1;
    379	buf[2] = 0;
    380
    381	err = irtoy_command(irtoy, buf, sizeof(buf), STATE_COMMAND_NO_RESP);
    382	if (err)
    383		dev_err(irtoy->dev, "could not write carrier command: %d\n",
    384			err);
    385
    386	return err;
    387}
    388
    389static int irtoy_probe(struct usb_interface *intf,
    390		       const struct usb_device_id *id)
    391{
    392	struct usb_host_interface *idesc = intf->cur_altsetting;
    393	struct usb_device *usbdev = interface_to_usbdev(intf);
    394	struct usb_endpoint_descriptor *ep_in = NULL;
    395	struct usb_endpoint_descriptor *ep_out = NULL;
    396	struct usb_endpoint_descriptor *ep = NULL;
    397	struct irtoy *irtoy;
    398	struct rc_dev *rc;
    399	struct urb *urb;
    400	int i, pipe, err = -ENOMEM;
    401
    402	for (i = 0; i < idesc->desc.bNumEndpoints; i++) {
    403		ep = &idesc->endpoint[i].desc;
    404
    405		if (!ep_in && usb_endpoint_is_bulk_in(ep) &&
    406		    usb_endpoint_maxp(ep) == MAX_PACKET)
    407			ep_in = ep;
    408
    409		if (!ep_out && usb_endpoint_is_bulk_out(ep) &&
    410		    usb_endpoint_maxp(ep) == MAX_PACKET)
    411			ep_out = ep;
    412	}
    413
    414	if (!ep_in || !ep_out) {
    415		dev_err(&intf->dev, "required endpoints not found\n");
    416		return -ENODEV;
    417	}
    418
    419	irtoy = kzalloc(sizeof(*irtoy), GFP_KERNEL);
    420	if (!irtoy)
    421		return -ENOMEM;
    422
    423	irtoy->in = kmalloc(MAX_PACKET,  GFP_KERNEL);
    424	if (!irtoy->in)
    425		goto free_irtoy;
    426
    427	irtoy->out = kmalloc(MAX_PACKET,  GFP_KERNEL);
    428	if (!irtoy->out)
    429		goto free_irtoy;
    430
    431	rc = rc_allocate_device(RC_DRIVER_IR_RAW);
    432	if (!rc)
    433		goto free_irtoy;
    434
    435	urb = usb_alloc_urb(0, GFP_KERNEL);
    436	if (!urb)
    437		goto free_rcdev;
    438
    439	pipe = usb_rcvbulkpipe(usbdev, ep_in->bEndpointAddress);
    440	usb_fill_bulk_urb(urb, usbdev, pipe, irtoy->in, MAX_PACKET,
    441			  irtoy_in_callback, irtoy);
    442	irtoy->urb_in = urb;
    443
    444	urb = usb_alloc_urb(0, GFP_KERNEL);
    445	if (!urb)
    446		goto free_rcdev;
    447
    448	pipe = usb_sndbulkpipe(usbdev, ep_out->bEndpointAddress);
    449	usb_fill_bulk_urb(urb, usbdev, pipe, irtoy->out, MAX_PACKET,
    450			  irtoy_out_callback, irtoy);
    451
    452	irtoy->dev = &intf->dev;
    453	irtoy->usbdev = usbdev;
    454	irtoy->rc = rc;
    455	irtoy->urb_out = urb;
    456	irtoy->pulse = true;
    457
    458	err = usb_submit_urb(irtoy->urb_in, GFP_KERNEL);
    459	if (err != 0) {
    460		dev_err(irtoy->dev, "fail to submit in urb: %d\n", err);
    461		goto free_rcdev;
    462	}
    463
    464	err = irtoy_setup(irtoy);
    465	if (err)
    466		goto free_rcdev;
    467
    468	dev_info(irtoy->dev, "version: hardware %u, firmware %u.%u, protocol %u",
    469		 irtoy->hw_version, irtoy->sw_version / 10,
    470		 irtoy->sw_version % 10, irtoy->proto_version);
    471
    472	if (irtoy->sw_version < MIN_FW_VERSION) {
    473		dev_err(irtoy->dev, "need firmware V%02u or higher",
    474			MIN_FW_VERSION);
    475		err = -ENODEV;
    476		goto free_rcdev;
    477	}
    478
    479	usb_make_path(usbdev, irtoy->phys, sizeof(irtoy->phys));
    480
    481	rc->device_name = "Infrared Toy";
    482	rc->driver_name = KBUILD_MODNAME;
    483	rc->input_phys = irtoy->phys;
    484	usb_to_input_id(usbdev, &rc->input_id);
    485	rc->dev.parent = &intf->dev;
    486	rc->priv = irtoy;
    487	rc->tx_ir = irtoy_tx;
    488	rc->s_tx_carrier = irtoy_tx_carrier;
    489	rc->allowed_protocols = RC_PROTO_BIT_ALL_IR_DECODER;
    490	rc->map_name = RC_MAP_RC6_MCE;
    491	rc->rx_resolution = UNIT_US;
    492	rc->timeout = IR_DEFAULT_TIMEOUT;
    493
    494	/*
    495	 * end of transmission is detected by absence of a usb packet
    496	 * with more pulse/spaces. However, each usb packet sent can
    497	 * contain 32 pulse/spaces, which can be quite lengthy, so there
    498	 * can be a delay between usb packets. For example with nec there is a
    499	 * 17ms gap between packets.
    500	 *
    501	 * So, make timeout a largish minimum which works with most protocols.
    502	 */
    503	rc->min_timeout = MS_TO_US(40);
    504	rc->max_timeout = MAX_TIMEOUT_US;
    505
    506	err = rc_register_device(rc);
    507	if (err)
    508		goto free_rcdev;
    509
    510	usb_set_intfdata(intf, irtoy);
    511
    512	return 0;
    513
    514free_rcdev:
    515	usb_kill_urb(irtoy->urb_out);
    516	usb_free_urb(irtoy->urb_out);
    517	usb_kill_urb(irtoy->urb_in);
    518	usb_free_urb(irtoy->urb_in);
    519	rc_free_device(rc);
    520free_irtoy:
    521	kfree(irtoy->in);
    522	kfree(irtoy->out);
    523	kfree(irtoy);
    524	return err;
    525}
    526
    527static void irtoy_disconnect(struct usb_interface *intf)
    528{
    529	struct irtoy *ir = usb_get_intfdata(intf);
    530
    531	rc_unregister_device(ir->rc);
    532	usb_set_intfdata(intf, NULL);
    533	usb_kill_urb(ir->urb_out);
    534	usb_free_urb(ir->urb_out);
    535	usb_kill_urb(ir->urb_in);
    536	usb_free_urb(ir->urb_in);
    537	kfree(ir->in);
    538	kfree(ir->out);
    539	kfree(ir);
    540}
    541
    542static const struct usb_device_id irtoy_table[] = {
    543	{ USB_DEVICE_INTERFACE_CLASS(0x04d8, 0xfd08, USB_CLASS_CDC_DATA) },
    544	{ USB_DEVICE_INTERFACE_CLASS(0x04d8, 0xf58b, USB_CLASS_CDC_DATA) },
    545	{ }
    546};
    547
    548static struct usb_driver irtoy_driver = {
    549	.name = KBUILD_MODNAME,
    550	.probe = irtoy_probe,
    551	.disconnect = irtoy_disconnect,
    552	.id_table = irtoy_table,
    553};
    554
    555module_usb_driver(irtoy_driver);
    556
    557MODULE_AUTHOR("Sean Young <sean@mess.org>");
    558MODULE_DESCRIPTION("Infrared Toy and IR Droid driver");
    559MODULE_LICENSE("GPL");
    560MODULE_DEVICE_TABLE(usb, irtoy_table);