| // SPDX-License-Identifier: GPL-2.0-only |
| /* |
| * Intel La Jolla Cove Adapter USB driver |
| * |
| * Copyright (c) 2023, Intel Corporation. |
| */ |
| |
| #include <linux/acpi.h> |
| #include <linux/auxiliary_bus.h> |
| #include <linux/dev_printk.h> |
| #include <linux/kernel.h> |
| #include <linux/mod_devicetable.h> |
| #include <linux/module.h> |
| #include <linux/mutex.h> |
| #include <linux/slab.h> |
| #include <linux/spinlock.h> |
| #include <linux/types.h> |
| #include <linux/usb.h> |
| #include <linux/usb/ljca.h> |
| |
| #include <asm/unaligned.h> |
| |
| /* command flags */ |
| #define LJCA_ACK_FLAG BIT(0) |
| #define LJCA_RESP_FLAG BIT(1) |
| #define LJCA_CMPL_FLAG BIT(2) |
| |
| #define LJCA_MAX_PACKET_SIZE 64u |
| #define LJCA_MAX_PAYLOAD_SIZE \ |
| (LJCA_MAX_PACKET_SIZE - sizeof(struct ljca_msg)) |
| |
| #define LJCA_WRITE_TIMEOUT_MS 200 |
| #define LJCA_WRITE_ACK_TIMEOUT_MS 500 |
| #define LJCA_ENUM_CLIENT_TIMEOUT_MS 20 |
| |
| /* ljca client type */ |
| enum ljca_client_type { |
| LJCA_CLIENT_MNG = 1, |
| LJCA_CLIENT_GPIO = 3, |
| LJCA_CLIENT_I2C = 4, |
| LJCA_CLIENT_SPI = 5, |
| }; |
| |
| /* MNG client commands */ |
| enum ljca_mng_cmd { |
| LJCA_MNG_RESET = 2, |
| LJCA_MNG_ENUM_GPIO = 4, |
| LJCA_MNG_ENUM_I2C = 5, |
| LJCA_MNG_ENUM_SPI = 8, |
| }; |
| |
| /* ljca client acpi _ADR */ |
| enum ljca_client_acpi_adr { |
| LJCA_GPIO_ACPI_ADR, |
| LJCA_I2C1_ACPI_ADR, |
| LJCA_I2C2_ACPI_ADR, |
| LJCA_SPI1_ACPI_ADR, |
| LJCA_SPI2_ACPI_ADR, |
| LJCA_CLIENT_ACPI_ADR_MAX, |
| }; |
| |
| /* ljca cmd message structure */ |
| struct ljca_msg { |
| u8 type; |
| u8 cmd; |
| u8 flags; |
| u8 len; |
| u8 data[] __counted_by(len); |
| } __packed; |
| |
| struct ljca_i2c_ctr_info { |
| u8 id; |
| u8 capacity; |
| u8 intr_pin; |
| } __packed; |
| |
| struct ljca_i2c_descriptor { |
| u8 num; |
| struct ljca_i2c_ctr_info info[] __counted_by(num); |
| } __packed; |
| |
| struct ljca_spi_ctr_info { |
| u8 id; |
| u8 capacity; |
| u8 intr_pin; |
| } __packed; |
| |
| struct ljca_spi_descriptor { |
| u8 num; |
| struct ljca_spi_ctr_info info[] __counted_by(num); |
| } __packed; |
| |
| struct ljca_bank_descriptor { |
| u8 bank_id; |
| u8 pin_num; |
| |
| /* 1 bit for each gpio, 1 means valid */ |
| __le32 valid_pins; |
| } __packed; |
| |
| struct ljca_gpio_descriptor { |
| u8 pins_per_bank; |
| u8 bank_num; |
| struct ljca_bank_descriptor bank_desc[] __counted_by(bank_num); |
| } __packed; |
| |
| /** |
| * struct ljca_adapter - represent a ljca adapter |
| * |
| * @intf: the usb interface for this ljca adapter |
| * @usb_dev: the usb device for this ljca adapter |
| * @dev: the specific device info of the usb interface |
| * @rx_pipe: bulk in pipe for receive data from firmware |
| * @tx_pipe: bulk out pipe for send data to firmware |
| * @rx_urb: urb used for the bulk in pipe |
| * @rx_buf: buffer used to receive command response and event |
| * @rx_len: length of rx buffer |
| * @ex_buf: external buffer to save command response |
| * @ex_buf_len: length of external buffer |
| * @actual_length: actual length of data copied to external buffer |
| * @tx_buf: buffer used to download command to firmware |
| * @tx_buf_len: length of tx buffer |
| * @lock: spinlock to protect tx_buf and ex_buf |
| * @cmd_completion: completion object as the command receives ack |
| * @mutex: mutex to avoid command download concurrently |
| * @client_list: client device list |
| * @disconnect: usb disconnect ongoing or not |
| * @reset_id: used to reset firmware |
| */ |
| struct ljca_adapter { |
| struct usb_interface *intf; |
| struct usb_device *usb_dev; |
| struct device *dev; |
| |
| unsigned int rx_pipe; |
| unsigned int tx_pipe; |
| |
| struct urb *rx_urb; |
| void *rx_buf; |
| unsigned int rx_len; |
| |
| u8 *ex_buf; |
| u8 ex_buf_len; |
| u8 actual_length; |
| |
| void *tx_buf; |
| u8 tx_buf_len; |
| |
| spinlock_t lock; |
| |
| struct completion cmd_completion; |
| struct mutex mutex; |
| |
| struct list_head client_list; |
| |
| bool disconnect; |
| |
| u32 reset_id; |
| }; |
| |
| struct ljca_match_ids_walk_data { |
| const struct acpi_device_id *ids; |
| const char *uid; |
| struct acpi_device *adev; |
| }; |
| |
| static const struct acpi_device_id ljca_gpio_hids[] = { |
| { "INTC1074" }, |
| { "INTC1096" }, |
| { "INTC100B" }, |
| { "INTC10D1" }, |
| { "INTC10B5" }, |
| {}, |
| }; |
| |
| static const struct acpi_device_id ljca_i2c_hids[] = { |
| { "INTC1075" }, |
| { "INTC1097" }, |
| { "INTC100C" }, |
| { "INTC10D2" }, |
| {}, |
| }; |
| |
| static const struct acpi_device_id ljca_spi_hids[] = { |
| { "INTC1091" }, |
| { "INTC1098" }, |
| { "INTC100D" }, |
| { "INTC10D3" }, |
| {}, |
| }; |
| |
| static void ljca_handle_event(struct ljca_adapter *adap, |
| struct ljca_msg *header) |
| { |
| struct ljca_client *client; |
| |
| list_for_each_entry(client, &adap->client_list, link) { |
| /* |
| * Currently only GPIO register event callback, but |
| * firmware message structure should include id when |
| * multiple same type clients register event callback. |
| */ |
| if (client->type == header->type) { |
| unsigned long flags; |
| |
| spin_lock_irqsave(&client->event_cb_lock, flags); |
| client->event_cb(client->context, header->cmd, |
| header->data, header->len); |
| spin_unlock_irqrestore(&client->event_cb_lock, flags); |
| |
| break; |
| } |
| } |
| } |
| |
| /* process command ack and received data if available */ |
| static void ljca_handle_cmd_ack(struct ljca_adapter *adap, struct ljca_msg *header) |
| { |
| struct ljca_msg *tx_header = adap->tx_buf; |
| u8 ibuf_len, actual_len = 0; |
| unsigned long flags; |
| u8 *ibuf; |
| |
| spin_lock_irqsave(&adap->lock, flags); |
| |
| if (tx_header->type != header->type || tx_header->cmd != header->cmd) { |
| spin_unlock_irqrestore(&adap->lock, flags); |
| dev_err(adap->dev, "cmd ack mismatch error\n"); |
| return; |
| } |
| |
| ibuf_len = adap->ex_buf_len; |
| ibuf = adap->ex_buf; |
| |
| if (ibuf && ibuf_len) { |
| actual_len = min(header->len, ibuf_len); |
| |
| /* copy received data to external buffer */ |
| memcpy(ibuf, header->data, actual_len); |
| } |
| /* update copied data length */ |
| adap->actual_length = actual_len; |
| |
| spin_unlock_irqrestore(&adap->lock, flags); |
| |
| complete(&adap->cmd_completion); |
| } |
| |
| static void ljca_recv(struct urb *urb) |
| { |
| struct ljca_msg *header = urb->transfer_buffer; |
| struct ljca_adapter *adap = urb->context; |
| int ret; |
| |
| switch (urb->status) { |
| case 0: |
| /* success */ |
| break; |
| case -ENOENT: |
| /* |
| * directly complete the possible ongoing transfer |
| * during disconnect |
| */ |
| if (adap->disconnect) |
| complete(&adap->cmd_completion); |
| return; |
| case -ECONNRESET: |
| case -ESHUTDOWN: |
| case -EPIPE: |
| /* rx urb is terminated */ |
| dev_dbg(adap->dev, "rx urb terminated with status: %d\n", |
| urb->status); |
| return; |
| default: |
| dev_dbg(adap->dev, "rx urb error: %d\n", urb->status); |
| goto resubmit; |
| } |
| |
| if (header->len + sizeof(*header) != urb->actual_length) |
| goto resubmit; |
| |
| if (header->flags & LJCA_ACK_FLAG) |
| ljca_handle_cmd_ack(adap, header); |
| else |
| ljca_handle_event(adap, header); |
| |
| resubmit: |
| ret = usb_submit_urb(urb, GFP_ATOMIC); |
| if (ret && ret != -EPERM) |
| dev_err(adap->dev, "resubmit rx urb error %d\n", ret); |
| } |
| |
| static int ljca_send(struct ljca_adapter *adap, u8 type, u8 cmd, |
| const u8 *obuf, u8 obuf_len, u8 *ibuf, u8 ibuf_len, |
| bool ack, unsigned long timeout) |
| { |
| unsigned int msg_len = sizeof(struct ljca_msg) + obuf_len; |
| struct ljca_msg *header = adap->tx_buf; |
| unsigned int transferred; |
| unsigned long flags; |
| int ret; |
| |
| if (adap->disconnect) |
| return -ENODEV; |
| |
| if (msg_len > adap->tx_buf_len) |
| return -EINVAL; |
| |
| mutex_lock(&adap->mutex); |
| |
| spin_lock_irqsave(&adap->lock, flags); |
| |
| header->type = type; |
| header->cmd = cmd; |
| header->len = obuf_len; |
| if (obuf) |
| memcpy(header->data, obuf, obuf_len); |
| |
| header->flags = LJCA_CMPL_FLAG | (ack ? LJCA_ACK_FLAG : 0); |
| |
| adap->ex_buf = ibuf; |
| adap->ex_buf_len = ibuf_len; |
| adap->actual_length = 0; |
| |
| spin_unlock_irqrestore(&adap->lock, flags); |
| |
| reinit_completion(&adap->cmd_completion); |
| |
| ret = usb_autopm_get_interface(adap->intf); |
| if (ret < 0) |
| goto out; |
| |
| ret = usb_bulk_msg(adap->usb_dev, adap->tx_pipe, header, |
| msg_len, &transferred, LJCA_WRITE_TIMEOUT_MS); |
| |
| usb_autopm_put_interface(adap->intf); |
| |
| if (ret < 0) |
| goto out; |
| if (transferred != msg_len) { |
| ret = -EIO; |
| goto out; |
| } |
| |
| if (ack) { |
| ret = wait_for_completion_timeout(&adap->cmd_completion, |
| timeout); |
| if (!ret) { |
| ret = -ETIMEDOUT; |
| goto out; |
| } |
| } |
| ret = adap->actual_length; |
| |
| out: |
| spin_lock_irqsave(&adap->lock, flags); |
| adap->ex_buf = NULL; |
| adap->ex_buf_len = 0; |
| |
| memset(header, 0, sizeof(*header)); |
| spin_unlock_irqrestore(&adap->lock, flags); |
| |
| mutex_unlock(&adap->mutex); |
| |
| return ret; |
| } |
| |
| int ljca_transfer(struct ljca_client *client, u8 cmd, const u8 *obuf, |
| u8 obuf_len, u8 *ibuf, u8 ibuf_len) |
| { |
| return ljca_send(client->adapter, client->type, cmd, |
| obuf, obuf_len, ibuf, ibuf_len, true, |
| LJCA_WRITE_ACK_TIMEOUT_MS); |
| } |
| EXPORT_SYMBOL_NS_GPL(ljca_transfer, LJCA); |
| |
| int ljca_transfer_noack(struct ljca_client *client, u8 cmd, const u8 *obuf, |
| u8 obuf_len) |
| { |
| return ljca_send(client->adapter, client->type, cmd, obuf, |
| obuf_len, NULL, 0, false, LJCA_WRITE_ACK_TIMEOUT_MS); |
| } |
| EXPORT_SYMBOL_NS_GPL(ljca_transfer_noack, LJCA); |
| |
| int ljca_register_event_cb(struct ljca_client *client, ljca_event_cb_t event_cb, |
| void *context) |
| { |
| unsigned long flags; |
| |
| if (!event_cb) |
| return -EINVAL; |
| |
| spin_lock_irqsave(&client->event_cb_lock, flags); |
| |
| if (client->event_cb) { |
| spin_unlock_irqrestore(&client->event_cb_lock, flags); |
| return -EALREADY; |
| } |
| |
| client->event_cb = event_cb; |
| client->context = context; |
| |
| spin_unlock_irqrestore(&client->event_cb_lock, flags); |
| |
| return 0; |
| } |
| EXPORT_SYMBOL_NS_GPL(ljca_register_event_cb, LJCA); |
| |
| void ljca_unregister_event_cb(struct ljca_client *client) |
| { |
| unsigned long flags; |
| |
| spin_lock_irqsave(&client->event_cb_lock, flags); |
| |
| client->event_cb = NULL; |
| client->context = NULL; |
| |
| spin_unlock_irqrestore(&client->event_cb_lock, flags); |
| } |
| EXPORT_SYMBOL_NS_GPL(ljca_unregister_event_cb, LJCA); |
| |
| static int ljca_match_device_ids(struct acpi_device *adev, void *data) |
| { |
| struct ljca_match_ids_walk_data *wd = data; |
| const char *uid = acpi_device_uid(adev); |
| |
| if (acpi_match_device_ids(adev, wd->ids)) |
| return 0; |
| |
| if (!wd->uid) |
| goto match; |
| |
| if (!uid) |
| /* |
| * Some DSDTs have only one ACPI companion for the two I2C |
| * controllers and they don't set a UID at all (e.g. Dell |
| * Latitude 9420). On these platforms only the first I2C |
| * controller is used, so if a HID match has no UID we use |
| * "0" as the UID and assign ACPI companion to the first |
| * I2C controller. |
| */ |
| uid = "0"; |
| else |
| uid = strchr(uid, wd->uid[0]); |
| |
| if (!uid || strcmp(uid, wd->uid)) |
| return 0; |
| |
| match: |
| wd->adev = adev; |
| |
| return 1; |
| } |
| |
| /* bind auxiliary device to acpi device */ |
| static void ljca_auxdev_acpi_bind(struct ljca_adapter *adap, |
| struct auxiliary_device *auxdev, |
| u64 adr, u8 id) |
| { |
| struct ljca_match_ids_walk_data wd = { 0 }; |
| struct device *dev = adap->dev; |
| struct acpi_device *parent; |
| char uid[4]; |
| |
| parent = ACPI_COMPANION(dev); |
| if (!parent) |
| return; |
| |
| /* |
| * Currently LJCA hw doesn't use _ADR instead the shipped |
| * platforms use _HID to distinguish children devices. |
| */ |
| switch (adr) { |
| case LJCA_GPIO_ACPI_ADR: |
| wd.ids = ljca_gpio_hids; |
| break; |
| case LJCA_I2C1_ACPI_ADR: |
| case LJCA_I2C2_ACPI_ADR: |
| snprintf(uid, sizeof(uid), "%d", id); |
| wd.uid = uid; |
| wd.ids = ljca_i2c_hids; |
| break; |
| case LJCA_SPI1_ACPI_ADR: |
| case LJCA_SPI2_ACPI_ADR: |
| wd.ids = ljca_spi_hids; |
| break; |
| default: |
| dev_warn(dev, "unsupported _ADR\n"); |
| return; |
| } |
| |
| acpi_dev_for_each_child(parent, ljca_match_device_ids, &wd); |
| if (wd.adev) { |
| ACPI_COMPANION_SET(&auxdev->dev, wd.adev); |
| return; |
| } |
| |
| parent = ACPI_COMPANION(dev->parent->parent); |
| if (!parent) |
| return; |
| |
| acpi_dev_for_each_child(parent, ljca_match_device_ids, &wd); |
| if (wd.adev) |
| ACPI_COMPANION_SET(&auxdev->dev, wd.adev); |
| } |
| |
| static void ljca_auxdev_release(struct device *dev) |
| { |
| struct auxiliary_device *auxdev = to_auxiliary_dev(dev); |
| |
| kfree(auxdev->dev.platform_data); |
| } |
| |
| static int ljca_new_client_device(struct ljca_adapter *adap, u8 type, u8 id, |
| char *name, void *data, u64 adr) |
| { |
| struct auxiliary_device *auxdev; |
| struct ljca_client *client; |
| int ret; |
| |
| client = kzalloc(sizeof *client, GFP_KERNEL); |
| if (!client) { |
| kfree(data); |
| return -ENOMEM; |
| } |
| |
| client->type = type; |
| client->id = id; |
| client->adapter = adap; |
| spin_lock_init(&client->event_cb_lock); |
| |
| auxdev = &client->auxdev; |
| auxdev->name = name; |
| auxdev->id = id; |
| |
| auxdev->dev.parent = adap->dev; |
| auxdev->dev.platform_data = data; |
| auxdev->dev.release = ljca_auxdev_release; |
| |
| ret = auxiliary_device_init(auxdev); |
| if (ret) { |
| kfree(data); |
| goto err_free; |
| } |
| |
| ljca_auxdev_acpi_bind(adap, auxdev, adr, id); |
| |
| ret = auxiliary_device_add(auxdev); |
| if (ret) |
| goto err_uninit; |
| |
| list_add_tail(&client->link, &adap->client_list); |
| |
| return 0; |
| |
| err_uninit: |
| auxiliary_device_uninit(auxdev); |
| |
| err_free: |
| kfree(client); |
| |
| return ret; |
| } |
| |
| static int ljca_enumerate_gpio(struct ljca_adapter *adap) |
| { |
| u32 valid_pin[LJCA_MAX_GPIO_NUM / BITS_PER_TYPE(u32)]; |
| struct ljca_gpio_descriptor *desc; |
| struct ljca_gpio_info *gpio_info; |
| u8 buf[LJCA_MAX_PAYLOAD_SIZE]; |
| int ret, gpio_num; |
| unsigned int i; |
| |
| ret = ljca_send(adap, LJCA_CLIENT_MNG, LJCA_MNG_ENUM_GPIO, NULL, 0, buf, |
| sizeof(buf), true, LJCA_ENUM_CLIENT_TIMEOUT_MS); |
| if (ret < 0) |
| return ret; |
| |
| /* check firmware response */ |
| desc = (struct ljca_gpio_descriptor *)buf; |
| if (ret != struct_size(desc, bank_desc, desc->bank_num)) |
| return -EINVAL; |
| |
| gpio_num = desc->pins_per_bank * desc->bank_num; |
| if (gpio_num > LJCA_MAX_GPIO_NUM) |
| return -EINVAL; |
| |
| /* construct platform data */ |
| gpio_info = kzalloc(sizeof *gpio_info, GFP_KERNEL); |
| if (!gpio_info) |
| return -ENOMEM; |
| gpio_info->num = gpio_num; |
| |
| for (i = 0; i < desc->bank_num; i++) |
| valid_pin[i] = get_unaligned_le32(&desc->bank_desc[i].valid_pins); |
| bitmap_from_arr32(gpio_info->valid_pin_map, valid_pin, gpio_num); |
| |
| return ljca_new_client_device(adap, LJCA_CLIENT_GPIO, 0, "ljca-gpio", |
| gpio_info, LJCA_GPIO_ACPI_ADR); |
| } |
| |
| static int ljca_enumerate_i2c(struct ljca_adapter *adap) |
| { |
| struct ljca_i2c_descriptor *desc; |
| struct ljca_i2c_info *i2c_info; |
| u8 buf[LJCA_MAX_PAYLOAD_SIZE]; |
| unsigned int i; |
| int ret; |
| |
| ret = ljca_send(adap, LJCA_CLIENT_MNG, LJCA_MNG_ENUM_I2C, NULL, 0, buf, |
| sizeof(buf), true, LJCA_ENUM_CLIENT_TIMEOUT_MS); |
| if (ret < 0) |
| return ret; |
| |
| /* check firmware response */ |
| desc = (struct ljca_i2c_descriptor *)buf; |
| if (ret != struct_size(desc, info, desc->num)) |
| return -EINVAL; |
| |
| for (i = 0; i < desc->num; i++) { |
| /* construct platform data */ |
| i2c_info = kzalloc(sizeof *i2c_info, GFP_KERNEL); |
| if (!i2c_info) |
| return -ENOMEM; |
| |
| i2c_info->id = desc->info[i].id; |
| i2c_info->capacity = desc->info[i].capacity; |
| i2c_info->intr_pin = desc->info[i].intr_pin; |
| |
| ret = ljca_new_client_device(adap, LJCA_CLIENT_I2C, i, |
| "ljca-i2c", i2c_info, |
| LJCA_I2C1_ACPI_ADR + i); |
| if (ret) |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static int ljca_enumerate_spi(struct ljca_adapter *adap) |
| { |
| struct ljca_spi_descriptor *desc; |
| struct ljca_spi_info *spi_info; |
| u8 buf[LJCA_MAX_PAYLOAD_SIZE]; |
| unsigned int i; |
| int ret; |
| |
| /* Not all LJCA chips implement SPI, a timeout reading the descriptors is normal */ |
| ret = ljca_send(adap, LJCA_CLIENT_MNG, LJCA_MNG_ENUM_SPI, NULL, 0, buf, |
| sizeof(buf), true, LJCA_ENUM_CLIENT_TIMEOUT_MS); |
| if (ret < 0) |
| return (ret == -ETIMEDOUT) ? 0 : ret; |
| |
| /* check firmware response */ |
| desc = (struct ljca_spi_descriptor *)buf; |
| if (ret != struct_size(desc, info, desc->num)) |
| return -EINVAL; |
| |
| for (i = 0; i < desc->num; i++) { |
| /* construct platform data */ |
| spi_info = kzalloc(sizeof *spi_info, GFP_KERNEL); |
| if (!spi_info) |
| return -ENOMEM; |
| |
| spi_info->id = desc->info[i].id; |
| spi_info->capacity = desc->info[i].capacity; |
| |
| ret = ljca_new_client_device(adap, LJCA_CLIENT_SPI, i, |
| "ljca-spi", spi_info, |
| LJCA_SPI1_ACPI_ADR + i); |
| if (ret) |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static int ljca_reset_handshake(struct ljca_adapter *adap) |
| { |
| __le32 reset_id = cpu_to_le32(adap->reset_id); |
| __le32 reset_id_ret = 0; |
| int ret; |
| |
| adap->reset_id++; |
| |
| ret = ljca_send(adap, LJCA_CLIENT_MNG, LJCA_MNG_RESET, (u8 *)&reset_id, |
| sizeof(__le32), (u8 *)&reset_id_ret, sizeof(__le32), |
| true, LJCA_WRITE_ACK_TIMEOUT_MS); |
| if (ret < 0) |
| return ret; |
| |
| if (reset_id_ret != reset_id) |
| return -EINVAL; |
| |
| return 0; |
| } |
| |
| static int ljca_enumerate_clients(struct ljca_adapter *adap) |
| { |
| struct ljca_client *client, *next; |
| int ret; |
| |
| ret = ljca_reset_handshake(adap); |
| if (ret) |
| goto err_kill; |
| |
| ret = ljca_enumerate_gpio(adap); |
| if (ret) { |
| dev_err(adap->dev, "enumerate GPIO error\n"); |
| goto err_kill; |
| } |
| |
| ret = ljca_enumerate_i2c(adap); |
| if (ret) { |
| dev_err(adap->dev, "enumerate I2C error\n"); |
| goto err_kill; |
| } |
| |
| ret = ljca_enumerate_spi(adap); |
| if (ret) { |
| dev_err(adap->dev, "enumerate SPI error\n"); |
| goto err_kill; |
| } |
| |
| return 0; |
| |
| err_kill: |
| adap->disconnect = true; |
| |
| usb_kill_urb(adap->rx_urb); |
| |
| list_for_each_entry_safe_reverse(client, next, &adap->client_list, link) { |
| auxiliary_device_delete(&client->auxdev); |
| auxiliary_device_uninit(&client->auxdev); |
| |
| list_del_init(&client->link); |
| kfree(client); |
| } |
| |
| return ret; |
| } |
| |
| static int ljca_probe(struct usb_interface *interface, |
| const struct usb_device_id *id) |
| { |
| struct usb_device *usb_dev = interface_to_usbdev(interface); |
| struct usb_host_interface *alt = interface->cur_altsetting; |
| struct usb_endpoint_descriptor *ep_in, *ep_out; |
| struct device *dev = &interface->dev; |
| struct ljca_adapter *adap; |
| int ret; |
| |
| adap = devm_kzalloc(dev, sizeof(*adap), GFP_KERNEL); |
| if (!adap) |
| return -ENOMEM; |
| |
| /* separate tx buffer allocation for alignment */ |
| adap->tx_buf = devm_kzalloc(dev, LJCA_MAX_PACKET_SIZE, GFP_KERNEL); |
| if (!adap->tx_buf) |
| return -ENOMEM; |
| adap->tx_buf_len = LJCA_MAX_PACKET_SIZE; |
| |
| mutex_init(&adap->mutex); |
| spin_lock_init(&adap->lock); |
| init_completion(&adap->cmd_completion); |
| INIT_LIST_HEAD(&adap->client_list); |
| |
| adap->intf = usb_get_intf(interface); |
| adap->usb_dev = usb_dev; |
| adap->dev = dev; |
| |
| /* |
| * find the first bulk in and out endpoints. |
| * ignore any others. |
| */ |
| ret = usb_find_common_endpoints(alt, &ep_in, &ep_out, NULL, NULL); |
| if (ret) { |
| dev_err(dev, "bulk endpoints not found\n"); |
| goto err_put; |
| } |
| adap->rx_pipe = usb_rcvbulkpipe(usb_dev, usb_endpoint_num(ep_in)); |
| adap->tx_pipe = usb_sndbulkpipe(usb_dev, usb_endpoint_num(ep_out)); |
| |
| /* setup rx buffer */ |
| adap->rx_len = usb_endpoint_maxp(ep_in); |
| adap->rx_buf = devm_kzalloc(dev, adap->rx_len, GFP_KERNEL); |
| if (!adap->rx_buf) { |
| ret = -ENOMEM; |
| goto err_put; |
| } |
| |
| /* alloc rx urb */ |
| adap->rx_urb = usb_alloc_urb(0, GFP_KERNEL); |
| if (!adap->rx_urb) { |
| ret = -ENOMEM; |
| goto err_put; |
| } |
| usb_fill_bulk_urb(adap->rx_urb, usb_dev, adap->rx_pipe, |
| adap->rx_buf, adap->rx_len, ljca_recv, adap); |
| |
| usb_set_intfdata(interface, adap); |
| |
| /* submit rx urb before enumerate clients */ |
| ret = usb_submit_urb(adap->rx_urb, GFP_KERNEL); |
| if (ret) { |
| dev_err(dev, "submit rx urb failed: %d\n", ret); |
| goto err_free; |
| } |
| |
| ret = ljca_enumerate_clients(adap); |
| if (ret) |
| goto err_free; |
| |
| usb_enable_autosuspend(usb_dev); |
| |
| return 0; |
| |
| err_free: |
| usb_free_urb(adap->rx_urb); |
| |
| err_put: |
| usb_put_intf(adap->intf); |
| |
| mutex_destroy(&adap->mutex); |
| |
| return ret; |
| } |
| |
| static void ljca_disconnect(struct usb_interface *interface) |
| { |
| struct ljca_adapter *adap = usb_get_intfdata(interface); |
| struct ljca_client *client, *next; |
| |
| adap->disconnect = true; |
| |
| usb_kill_urb(adap->rx_urb); |
| |
| list_for_each_entry_safe_reverse(client, next, &adap->client_list, link) { |
| auxiliary_device_delete(&client->auxdev); |
| auxiliary_device_uninit(&client->auxdev); |
| |
| list_del_init(&client->link); |
| kfree(client); |
| } |
| |
| usb_free_urb(adap->rx_urb); |
| |
| usb_put_intf(adap->intf); |
| |
| mutex_destroy(&adap->mutex); |
| } |
| |
| static int ljca_suspend(struct usb_interface *interface, pm_message_t message) |
| { |
| struct ljca_adapter *adap = usb_get_intfdata(interface); |
| |
| usb_kill_urb(adap->rx_urb); |
| |
| return 0; |
| } |
| |
| static int ljca_resume(struct usb_interface *interface) |
| { |
| struct ljca_adapter *adap = usb_get_intfdata(interface); |
| |
| return usb_submit_urb(adap->rx_urb, GFP_KERNEL); |
| } |
| |
| static const struct usb_device_id ljca_table[] = { |
| { USB_DEVICE(0x8086, 0x0b63) }, |
| { /* sentinel */ } |
| }; |
| MODULE_DEVICE_TABLE(usb, ljca_table); |
| |
| static struct usb_driver ljca_driver = { |
| .name = "ljca", |
| .id_table = ljca_table, |
| .probe = ljca_probe, |
| .disconnect = ljca_disconnect, |
| .suspend = ljca_suspend, |
| .resume = ljca_resume, |
| .supports_autosuspend = 1, |
| }; |
| module_usb_driver(ljca_driver); |
| |
| MODULE_AUTHOR("Wentong Wu <wentong.wu@intel.com>"); |
| MODULE_AUTHOR("Zhifeng Wang <zhifeng.wang@intel.com>"); |
| MODULE_AUTHOR("Lixu Zhang <lixu.zhang@intel.com>"); |
| MODULE_DESCRIPTION("Intel La Jolla Cove Adapter USB driver"); |
| MODULE_LICENSE("GPL"); |