|
|
|
@@ -8,11 +8,14 @@
|
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
|
#include <linux/irqreturn.h>
|
|
|
|
|
#include <linux/pci.h>
|
|
|
|
|
#include <linux/sizes.h>
|
|
|
|
|
|
|
|
|
|
#include "intel-thc-dev.h"
|
|
|
|
|
#include "intel-thc-hw.h"
|
|
|
|
|
|
|
|
|
|
#include "quicki2c-dev.h"
|
|
|
|
|
#include "quicki2c-hid.h"
|
|
|
|
|
#include "quicki2c-protocol.h"
|
|
|
|
|
|
|
|
|
|
/* THC QuickI2C ACPI method to get device properties */
|
|
|
|
|
/* HIDI2C device method */
|
|
|
|
@@ -210,6 +213,69 @@ static irqreturn_t quicki2c_irq_quick_handler(int irq, void *dev_id)
|
|
|
|
|
return IRQ_WAKE_THREAD;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* try_recover - Try to recovery THC and Device
|
|
|
|
|
* @qcdev: pointer to quicki2c device
|
|
|
|
|
*
|
|
|
|
|
* This function is a error handler, called when fatal error happens.
|
|
|
|
|
* It try to reset Touch Device and re-configure THC to recovery
|
|
|
|
|
* transferring between Device and THC.
|
|
|
|
|
*
|
|
|
|
|
* Return: 0 if successful or error code on failed
|
|
|
|
|
*/
|
|
|
|
|
static int try_recover(struct quicki2c_device *qcdev)
|
|
|
|
|
{
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
thc_dma_unconfigure(qcdev->thc_hw);
|
|
|
|
|
|
|
|
|
|
ret = thc_dma_configure(qcdev->thc_hw);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(qcdev->dev, "Reconfig DMA failed\n");
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int handle_input_report(struct quicki2c_device *qcdev)
|
|
|
|
|
{
|
|
|
|
|
struct hidi2c_report_packet *pkt = (struct hidi2c_report_packet *)qcdev->input_buf;
|
|
|
|
|
int rx_dma_finished = 0;
|
|
|
|
|
size_t report_len;
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
while (!rx_dma_finished) {
|
|
|
|
|
ret = thc_rxdma_read(qcdev->thc_hw, THC_RXDMA2,
|
|
|
|
|
(u8 *)pkt, &report_len,
|
|
|
|
|
&rx_dma_finished);
|
|
|
|
|
if (ret)
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
|
|
if (!pkt->len) {
|
|
|
|
|
if (qcdev->state == QUICKI2C_RESETING) {
|
|
|
|
|
qcdev->reset_ack = true;
|
|
|
|
|
wake_up(&qcdev->reset_ack_wq);
|
|
|
|
|
|
|
|
|
|
qcdev->state = QUICKI2C_RESETED;
|
|
|
|
|
} else {
|
|
|
|
|
dev_warn(qcdev->dev, "unexpected DIR happen\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* discard samples before driver probe complete */
|
|
|
|
|
if (qcdev->state != QUICKI2C_ENABLED)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
quicki2c_hid_send_report(qcdev, pkt->data,
|
|
|
|
|
HIDI2C_DATA_LEN(le16_to_cpu(pkt->len)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* quicki2c_irq_thread_handler - IRQ thread handler of quicki2c driver
|
|
|
|
|
*
|
|
|
|
@@ -221,6 +287,7 @@ static irqreturn_t quicki2c_irq_quick_handler(int irq, void *dev_id)
|
|
|
|
|
static irqreturn_t quicki2c_irq_thread_handler(int irq, void *dev_id)
|
|
|
|
|
{
|
|
|
|
|
struct quicki2c_device *qcdev = dev_id;
|
|
|
|
|
int err_recover = 0;
|
|
|
|
|
int int_mask;
|
|
|
|
|
|
|
|
|
|
if (qcdev->state == QUICKI2C_DISABLED)
|
|
|
|
@@ -228,8 +295,25 @@ static irqreturn_t quicki2c_irq_thread_handler(int irq, void *dev_id)
|
|
|
|
|
|
|
|
|
|
int_mask = thc_interrupt_handler(qcdev->thc_hw);
|
|
|
|
|
|
|
|
|
|
if (int_mask & BIT(THC_FATAL_ERR_INT) || int_mask & BIT(THC_TXN_ERR_INT) ||
|
|
|
|
|
int_mask & BIT(THC_UNKNOWN_INT)) {
|
|
|
|
|
err_recover = 1;
|
|
|
|
|
goto exit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (int_mask & BIT(THC_RXDMA2_INT)) {
|
|
|
|
|
err_recover = handle_input_report(qcdev);
|
|
|
|
|
if (err_recover)
|
|
|
|
|
goto exit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
exit:
|
|
|
|
|
thc_interrupt_enable(qcdev->thc_hw, true);
|
|
|
|
|
|
|
|
|
|
if (err_recover)
|
|
|
|
|
if (try_recover(qcdev))
|
|
|
|
|
qcdev->state = QUICKI2C_DISABLED;
|
|
|
|
|
|
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@@ -260,6 +344,9 @@ static struct quicki2c_device *quicki2c_dev_init(struct pci_dev *pdev, void __io
|
|
|
|
|
qcdev->pdev = pdev;
|
|
|
|
|
qcdev->dev = dev;
|
|
|
|
|
qcdev->mem_addr = mem_addr;
|
|
|
|
|
qcdev->state = QUICKI2C_DISABLED;
|
|
|
|
|
|
|
|
|
|
init_waitqueue_head(&qcdev->reset_ack_wq);
|
|
|
|
|
|
|
|
|
|
/* thc hw init */
|
|
|
|
|
qcdev->thc_hw = thc_dev_init(qcdev->dev, qcdev->mem_addr);
|
|
|
|
@@ -275,6 +362,10 @@ static struct quicki2c_device *quicki2c_dev_init(struct pci_dev *pdev, void __io
|
|
|
|
|
return ERR_PTR(ret);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = thc_interrupt_quiesce(qcdev->thc_hw, true);
|
|
|
|
|
if (ret)
|
|
|
|
|
return ERR_PTR(ret);
|
|
|
|
|
|
|
|
|
|
ret = thc_port_select(qcdev->thc_hw, THC_PORT_TYPE_I2C);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err_once(dev, "Failed to select THC port, ret = %d.\n", ret);
|
|
|
|
@@ -288,10 +379,14 @@ static struct quicki2c_device *quicki2c_dev_init(struct pci_dev *pdev, void __io
|
|
|
|
|
if (ret)
|
|
|
|
|
return ERR_PTR(ret);
|
|
|
|
|
|
|
|
|
|
thc_int_trigger_type_select(qcdev->thc_hw, false);
|
|
|
|
|
|
|
|
|
|
thc_interrupt_config(qcdev->thc_hw);
|
|
|
|
|
|
|
|
|
|
thc_interrupt_enable(qcdev->thc_hw, true);
|
|
|
|
|
|
|
|
|
|
qcdev->state = QUICKI2C_INITED;
|
|
|
|
|
|
|
|
|
|
return qcdev;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@@ -305,6 +400,114 @@ static struct quicki2c_device *quicki2c_dev_init(struct pci_dev *pdev, void __io
|
|
|
|
|
static void quicki2c_dev_deinit(struct quicki2c_device *qcdev)
|
|
|
|
|
{
|
|
|
|
|
thc_interrupt_enable(qcdev->thc_hw, false);
|
|
|
|
|
thc_ltr_unconfig(qcdev->thc_hw);
|
|
|
|
|
|
|
|
|
|
qcdev->state = QUICKI2C_DISABLED;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* quicki2c_dma_init - Configure THC DMA for quicki2c device
|
|
|
|
|
* @qcdev: pointer to the quicki2c device structure
|
|
|
|
|
*
|
|
|
|
|
* This function uses TIC's parameters(such as max input length, max output
|
|
|
|
|
* length) to allocate THC DMA buffers and configure THC DMA engines.
|
|
|
|
|
*
|
|
|
|
|
* Return: 0 if success or error code on failed.
|
|
|
|
|
*/
|
|
|
|
|
static int quicki2c_dma_init(struct quicki2c_device *qcdev)
|
|
|
|
|
{
|
|
|
|
|
size_t swdma_max_len;
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
swdma_max_len = max(le16_to_cpu(qcdev->dev_desc.max_input_len),
|
|
|
|
|
le16_to_cpu(qcdev->dev_desc.report_desc_len));
|
|
|
|
|
|
|
|
|
|
ret = thc_dma_set_max_packet_sizes(qcdev->thc_hw, 0,
|
|
|
|
|
le16_to_cpu(qcdev->dev_desc.max_input_len),
|
|
|
|
|
le16_to_cpu(qcdev->dev_desc.max_output_len),
|
|
|
|
|
swdma_max_len);
|
|
|
|
|
if (ret)
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
|
|
ret = thc_dma_allocate(qcdev->thc_hw);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(qcdev->dev, "Allocate THC DMA buffer failed, ret = %d\n", ret);
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Enable RxDMA */
|
|
|
|
|
ret = thc_dma_configure(qcdev->thc_hw);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(qcdev->dev, "Configure THC DMA failed, ret = %d\n", ret);
|
|
|
|
|
thc_dma_unconfigure(qcdev->thc_hw);
|
|
|
|
|
thc_dma_release(qcdev->thc_hw);
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* quicki2c_dma_deinit - Release THC DMA for quicki2c device
|
|
|
|
|
* @qcdev: pointer to the quicki2c device structure
|
|
|
|
|
*
|
|
|
|
|
* Stop THC DMA engines and release all DMA buffers.
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
static void quicki2c_dma_deinit(struct quicki2c_device *qcdev)
|
|
|
|
|
{
|
|
|
|
|
thc_dma_unconfigure(qcdev->thc_hw);
|
|
|
|
|
thc_dma_release(qcdev->thc_hw);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* quicki2c_alloc_report_buf - Alloc report buffers
|
|
|
|
|
* @qcdev: pointer to the quicki2c device structure
|
|
|
|
|
*
|
|
|
|
|
* Allocate report descriptor buffer, it will be used for restore TIC HID
|
|
|
|
|
* report descriptor.
|
|
|
|
|
*
|
|
|
|
|
* Allocate input report buffer, it will be used for receive HID input report
|
|
|
|
|
* data from TIC.
|
|
|
|
|
*
|
|
|
|
|
* Allocate output report buffer, it will be used for store HID output report,
|
|
|
|
|
* such as set feature.
|
|
|
|
|
*
|
|
|
|
|
* Return: 0 if success or error code on failed.
|
|
|
|
|
*/
|
|
|
|
|
static int quicki2c_alloc_report_buf(struct quicki2c_device *qcdev)
|
|
|
|
|
{
|
|
|
|
|
size_t max_report_len;
|
|
|
|
|
|
|
|
|
|
qcdev->report_descriptor = devm_kzalloc(qcdev->dev,
|
|
|
|
|
le16_to_cpu(qcdev->dev_desc.report_desc_len),
|
|
|
|
|
GFP_KERNEL);
|
|
|
|
|
if (!qcdev->report_descriptor)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Some HIDI2C devices don't declare input/output max length correctly,
|
|
|
|
|
* give default 4K buffer to avoid DMA buffer overrun.
|
|
|
|
|
*/
|
|
|
|
|
max_report_len = max(le16_to_cpu(qcdev->dev_desc.max_input_len), SZ_4K);
|
|
|
|
|
|
|
|
|
|
qcdev->input_buf = devm_kzalloc(qcdev->dev, max_report_len, GFP_KERNEL);
|
|
|
|
|
if (!qcdev->input_buf)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
if (!le16_to_cpu(qcdev->dev_desc.max_output_len))
|
|
|
|
|
qcdev->dev_desc.max_output_len = cpu_to_le16(SZ_4K);
|
|
|
|
|
|
|
|
|
|
max_report_len = max(le16_to_cpu(qcdev->dev_desc.max_output_len),
|
|
|
|
|
max_report_len);
|
|
|
|
|
|
|
|
|
|
qcdev->report_buf = devm_kzalloc(qcdev->dev, max_report_len, GFP_KERNEL);
|
|
|
|
|
if (!qcdev->report_buf)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
qcdev->report_len = max_report_len;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
@@ -313,6 +516,18 @@ static void quicki2c_dev_deinit(struct quicki2c_device *qcdev)
|
|
|
|
|
* @pdev: point to pci device
|
|
|
|
|
* @id: point to pci_device_id structure
|
|
|
|
|
*
|
|
|
|
|
* This function initializes THC and HIDI2C device, the flow is:
|
|
|
|
|
* - do THC pci device initialization
|
|
|
|
|
* - query HIDI2C ACPI parameters
|
|
|
|
|
* - configure THC to HIDI2C mode
|
|
|
|
|
* - go through HIDI2C enumeration flow
|
|
|
|
|
* |- read device descriptor
|
|
|
|
|
* |- reset HIDI2C device
|
|
|
|
|
* - enable THC interrupt and DMA
|
|
|
|
|
* - read report descriptor
|
|
|
|
|
* - register HID device
|
|
|
|
|
* - enable runtime power management
|
|
|
|
|
*
|
|
|
|
|
* Return 0 if success or error code on failed.
|
|
|
|
|
*/
|
|
|
|
|
static int quicki2c_probe(struct pci_dev *pdev,
|
|
|
|
@@ -376,8 +591,60 @@ static int quicki2c_probe(struct pci_dev *pdev,
|
|
|
|
|
goto dev_deinit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = quicki2c_get_device_descriptor(qcdev);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(&pdev->dev, "Get device descriptor failed, ret = %d\n", ret);
|
|
|
|
|
goto dev_deinit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = quicki2c_alloc_report_buf(qcdev);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(&pdev->dev, "Alloc report buffers failed, ret= %d\n", ret);
|
|
|
|
|
goto dev_deinit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = quicki2c_dma_init(qcdev);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(&pdev->dev, "Setup THC DMA failed, ret= %d\n", ret);
|
|
|
|
|
goto dev_deinit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = thc_interrupt_quiesce(qcdev->thc_hw, false);
|
|
|
|
|
if (ret)
|
|
|
|
|
goto dev_deinit;
|
|
|
|
|
|
|
|
|
|
ret = quicki2c_set_power(qcdev, HIDI2C_ON);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(&pdev->dev, "Set Power On command failed, ret= %d\n", ret);
|
|
|
|
|
goto dev_deinit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = quicki2c_reset(qcdev);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(&pdev->dev, "Reset HIDI2C device failed, ret= %d\n", ret);
|
|
|
|
|
goto dev_deinit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = quicki2c_get_report_descriptor(qcdev);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(&pdev->dev, "Get report descriptor failed, ret = %d\n", ret);
|
|
|
|
|
goto dma_deinit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = quicki2c_hid_probe(qcdev);
|
|
|
|
|
if (ret) {
|
|
|
|
|
dev_err(&pdev->dev, "Failed to register HID device, ret = %d\n", ret);
|
|
|
|
|
goto dma_deinit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
qcdev->state = QUICKI2C_ENABLED;
|
|
|
|
|
|
|
|
|
|
dev_dbg(&pdev->dev, "QuickI2C probe success\n");
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
dma_deinit:
|
|
|
|
|
quicki2c_dma_deinit(qcdev);
|
|
|
|
|
dev_deinit:
|
|
|
|
|
quicki2c_dev_deinit(qcdev);
|
|
|
|
|
unmap_io_region:
|
|
|
|
@@ -404,6 +671,9 @@ static void quicki2c_remove(struct pci_dev *pdev)
|
|
|
|
|
if (!qcdev)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
quicki2c_hid_remove(qcdev);
|
|
|
|
|
quicki2c_dma_deinit(qcdev);
|
|
|
|
|
|
|
|
|
|
quicki2c_dev_deinit(qcdev);
|
|
|
|
|
|
|
|
|
|
pcim_iounmap_regions(pdev, BIT(0));
|
|
|
|
@@ -427,6 +697,9 @@ static void quicki2c_shutdown(struct pci_dev *pdev)
|
|
|
|
|
if (!qcdev)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/* Must stop DMA before reboot to avoid DMA entering into unknown state */
|
|
|
|
|
quicki2c_dma_deinit(qcdev);
|
|
|
|
|
|
|
|
|
|
quicki2c_dev_deinit(qcdev);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|