저는 Linux 우분투 5.4.0-1026-raspi를 사용하고 있습니다. Cypress CY8CKIT-059 PSoC 5LP용 USB 드라이버를 직접 작성하려고 합니다. Cypress PSoC의 USB 인터페이스는 대량 입력 엔드포인트를 통해 수신된 모든 데이터를 대량 출력 엔드포인트를 통해 다시 전송하도록 구성됩니다. 기본적으로 이 방법은 작동하지만 데이터 양이 적은 경우에만 가능합니다. 매직 배리어는 약 2000바이트 정도이다. 2000바이트 이상을 전송할 때 전체 시스템이 갑자기 중단되었습니다. 시스템 로그에서 유용한 정보를 찾을 수 없습니다. 누군가 여기서 무슨 일이 일어나고 있는지 나에게 설명해 줄 수 있습니까?
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/usb.h>
#include <linux/slab.h>
#include <linux/uaccess.h>
#include <linux/circ_buf.h>
#include <linux/kref.h>
#include <linux/mutex.h>
#include <linux/poll.h>
#include <linux/wait.h> /* wait_queue_head_t, wait_event_interruptible, wake_up_interruptible */
#define DRIVER_NAME "Cypress_USB_Bulk_Transfer_Driver"
#define USB_CYPRESS_VENDOR_ID 0x04B4
#define USB_CYPRESS_PRODUCT_ID 0x8051
#define MIN(a,b) (((a) <= (b)) ? (a) : (b))
#define KB *1024
#define RING_BUFFER_SIZE 8 KB
MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("Cypress USB Bulk Data Transfer Driver");
MODULE_VERSION("1.0");
/* static spinlock */
static DEFINE_SPINLOCK(lock);
/* read waitqueue */
static wait_queue_head_t read_waitqueue;
/* prototypes */
static int cypress_probe(struct usb_interface *intf, const struct usb_device_id *id);
static void cypress_disconnect(struct usb_interface *interface);
static int cypress_open(struct inode *inode, struct file *file);
static int cypress_release(struct inode *inode, struct file *file);
static ssize_t cypress_write(struct file *file, const char __user *buffer, size_t count, loff_t *offset);
static void cypress_write_bulk_callback(struct urb *urb);
static ssize_t cypress_read(struct file *file, char *buffer, size_t count, loff_t *offset);
static unsigned int cypress_poll(struct file *file, struct poll_table_struct *wait);
/* device specific structure */
struct cypress_usb_dev{
struct usb_device *device;
struct usb_host_interface *interface;
/* bulk in */
size_t bulk_in_size;
__u8 bulk_in_endpointAddr;
struct circ_buf bulk_in_ring;
/*bulk out */
size_t bulk_out_size;
__u8 bulk_out_endpointAddr;
char *bulk_out_buffer;
/* urbs */
struct urb *urb_r1, *urb_r2;
struct usb_anchor read_urbs, write_urbs;
/* buffer to receive data */
void *buffer_urb_in1;
dma_addr_t dma_buffer_urb_in1;
bool submitted_urb_r1;
void *buffer_urb_in2;
dma_addr_t dma_buffer_urb_in2;
bool submitted_urb_r2;
struct kref kref;
};
/* table of devices that work with this driver */
static struct usb_device_id id_table [] = {
{ USB_DEVICE(USB_CYPRESS_VENDOR_ID, USB_CYPRESS_PRODUCT_ID) },
{}, /* Terminating entry */
};
MODULE_DEVICE_TABLE (usb, id_table);
/* USB driver struct */
static struct usb_driver cypress_driver = {
.name = "Cypress USB Bulk Data Transfer driver",
.probe = cypress_probe,
.disconnect = cypress_disconnect,
.id_table = id_table,
};
/* file_operations struct callback for system calls */
static const struct file_operations cypress_fops = {
.owner = THIS_MODULE,
.open = cypress_open,
.read = cypress_read,
.write = cypress_write,
.poll = cypress_poll,
.release = cypress_release,
};
/* used to register for a minor number with the USB subsystem */
static struct usb_class_driver cypress_class = {
.name = "usb/cypress_driver%d", /* name shown in sysfs */
.fops = &cypress_fops,
};
static void cypress_delete(struct kref *kref)
{
struct cypress_usb_dev *cyp = container_of(kref, struct cypress_usb_dev, kref);
wake_up(&read_waitqueue);
/* free device memory */
usb_kill_anchored_urbs(&cyp->read_urbs);
//usb_kill_urb(cyp->urb_r1);
//usb_kill_urb(cyp->urb_r2);
usb_kill_anchored_urbs(&cyp->write_urbs);
usb_free_coherent(cyp->device, cyp->bulk_in_size, cyp->buffer_urb_in1, cyp->dma_buffer_urb_in1);
usb_free_coherent(cyp->device, cyp->bulk_in_size, cyp->buffer_urb_in2, cyp->dma_buffer_urb_in2);
kfree(cyp->bulk_in_ring.buf);
kfree(cyp);
}
/* probe function is called on device insertation through the USB core */
static int cypress_probe(struct usb_interface *interface, const struct usb_device_id *id)
{
struct cypress_usb_dev *cyp;
struct usb_device *device;
struct usb_host_interface *intf_desc;
struct usb_endpoint_descriptor *endpoint;
int i, minor, retval;
/* allocate memory for the device */
cyp = kzalloc(sizeof(struct cypress_usb_dev), GFP_KERNEL);
if(cyp == NULL)
{
printk(KERN_ALERT "%s Failed to allocate memory for the device.", DRIVER_NAME);
retval = -ENOMEM;
goto error;
}
kref_init(&cyp->kref);
init_usb_anchor(&cyp->read_urbs);
init_usb_anchor(&cyp->write_urbs);
init_waitqueue_head(&read_waitqueue);
/* get container device from interface pointer */
device = interface_to_usbdev(interface);
cyp->device = device;
/* get pointer to interface descriptor */
intf_desc = interface->cur_altsetting;
cyp->interface = intf_desc;
printk(KERN_INFO "%s [*] Cypress USB BULK Transfer device detected (Vendor:%04X Product:%04X)", DRIVER_NAME, id->idVendor, id->idProduct);
printk(KERN_INFO "%s Interface %d with class %02X and %d endpoints now probed", \
DRIVER_NAME, intf_desc->desc.bInterfaceNumber, intf_desc->desc.bInterfaceClass, intf_desc->desc.bNumEndpoints);
/* print all available endpoints */
for(i=0; i<intf_desc->desc.bNumEndpoints; i++)
{
/* get pointer to endpoint */
endpoint = &intf_desc->endpoint[i].desc;
/* print endpoint information */
printk(KERN_INFO "%s\t\t EP[%d]->bEndpointAddress: 0x%02X", DRIVER_NAME, i, endpoint->bEndpointAddress);
printk(KERN_INFO "%s\t\t EP[%d]->bDescriptorType: 0x%02X", DRIVER_NAME, i, endpoint->bDescriptorType);
printk(KERN_INFO "%s\t\t EP[%d]->wMaxPacketSize: 0x%04X (%d)", DRIVER_NAME, i, endpoint->wMaxPacketSize, endpoint->wMaxPacketSize);
}
/* check if bulk in endpoint is available in the probed interface */
if(!usb_find_bulk_in_endpoint(intf_desc, &endpoint))
{
/* store endpoint information */
cyp->bulk_in_size = endpoint->wMaxPacketSize;
cyp->bulk_in_endpointAddr = endpoint->bEndpointAddress;
printk(KERN_INFO "%s Found suitable bulk in endpoint in interface", DRIVER_NAME);
}
else
{
printk(KERN_ALERT "%s Failed to find suitable bulk in endpoint in interface", DRIVER_NAME);
retval = -ENODEV;
goto error;
}
/* check if bulk out endpoint is available in the probed interface */
if(!usb_find_bulk_out_endpoint(intf_desc, &endpoint))
{
/* store endpoint information */
cyp->bulk_out_size = endpoint->wMaxPacketSize;
cyp->bulk_out_endpointAddr = endpoint->bEndpointAddress;
printk(KERN_INFO "%s Found suitable bulk out endpoint in interface", DRIVER_NAME);
}
else
{
printk(KERN_ALERT "%s Failed to find suitable bulk out endpoint in interface", DRIVER_NAME);
retval = -ENODEV;
goto error;
}
/* register device and ask for a minor number */
minor = usb_register_dev(interface, &cypress_class);
if(minor)
{
printk(KERN_ALERT "%s Failed to register USB device. error: %d", DRIVER_NAME, minor);
usb_deregister_dev(interface, &cypress_class);
retval = minor;
goto error;
}
printk(KERN_INFO "%s Minor %d obtained for Cypress USB Bulk Transfer device", DRIVER_NAME, interface->minor);
/* save driver data within the interface */
usb_set_intfdata(interface, cyp);
return 0;
error:
kref_put(&cyp->kref, cypress_delete);
return retval;
}
/* callback function for read urb */
static void cypress_read_bulk_callback1(struct urb *urb)
{
struct cypress_usb_dev *cyp;
int retval;
unsigned long flags;
/* get pointer to device struct */
cyp = urb->context;
cyp->submitted_urb_r1 = false;
printk(KERN_DEBUG "%s URB1 read callback triggered %s", DRIVER_NAME, (char*)urb->transfer_buffer);
if (urb->status)
{
/* device was unlinked do not resubmit urb */
if(urb->status == -ENOENT || urb->status == -ECONNRESET || urb->status == -ESHUTDOWN)
{
return;
}
printk(KERN_ERR "%s Failed bulk read attemting to resubmit, error: %d", DRIVER_NAME, urb->status);
/* using spin_lock_irqsave - just because it is always save */
spin_lock_irqsave(&lock, flags);
goto resubmit;
}
/* using spin_lock_irqsave - just because it is always save */
spin_lock_irqsave(&lock, flags);
/* copy data to circular buffer */
memcpy(&cyp->bulk_in_ring.buf[cyp->bulk_in_ring.head], urb->transfer_buffer, urb->actual_length);
/* update circular buffer index */
cyp->bulk_in_ring.head = (cyp->bulk_in_ring.head + urb->actual_length) & (RING_BUFFER_SIZE -1);
wake_up(&read_waitqueue);
printk(KERN_NOTICE "%s Head at %d", DRIVER_NAME, cyp->bulk_in_ring.head);
resubmit:
/* check if enough space is available in circular buffer */
if(CIRC_SPACE(cyp->bulk_in_ring.head, cyp->bulk_in_ring.tail, RING_BUFFER_SIZE) < (cyp->urb_r1->transfer_buffer_length + cyp->urb_r2->transfer_buffer_length))
{
printk(KERN_NOTICE "%s Bulk in Circular Buffer full", DRIVER_NAME);
goto exit;
}
/* resubmit urb if enough space is available in circular buffer */
if ((retval = usb_submit_urb(urb, GFP_KERNEL))) {
printk(KERN_ERR "%s Failed to resubmit read URB, error %d", DRIVER_NAME, retval);
goto exit;
}
usb_anchor_urb(urb, &cyp->read_urbs);
cyp->submitted_urb_r1 = true;
exit:
spin_unlock_irqrestore(&lock, flags);
return;
}
/* callback function for read urb */
static void cypress_read_bulk_callback2(struct urb *urb)
{
struct cypress_usb_dev *cyp;
int retval;
unsigned long flags;
/* get pointer to device struct */
cyp = urb->context;
cyp->submitted_urb_r2 = false;
printk(KERN_DEBUG "%s URB2 read callback triggered %s", DRIVER_NAME, (char*)urb->transfer_buffer);
if (urb->status)
{
/* device was unlinked do not resubmit urb */
if(urb->status == -ENOENT || urb->status == -ECONNRESET || urb->status == -ESHUTDOWN)
{
return;
}
printk(KERN_ERR "%s Failed bulk read attemting to resubmit, error: %d", DRIVER_NAME, urb->status);
/* using spin_lock_irqsave - just because it is always save */
spin_lock_irqsave(&lock, flags);
goto resubmit;
}
/* using spin_lock_irqsave - just because it is always save */
spin_lock_irqsave(&lock, flags);
/* copy data to circular buffer */
memcpy(&cyp->bulk_in_ring.buf[cyp->bulk_in_ring.head], urb->transfer_buffer, urb->actual_length);
/* update circular buffer index */
cyp->bulk_in_ring.head = (cyp->bulk_in_ring.head + urb->actual_length) & (RING_BUFFER_SIZE -1);
wake_up(&read_waitqueue);
printk(KERN_NOTICE "%s Head at %d", DRIVER_NAME, cyp->bulk_in_ring.head);
resubmit:
/* check if enough space is available in circular buffer */
if(CIRC_SPACE(cyp->bulk_in_ring.head, cyp->bulk_in_ring.tail, RING_BUFFER_SIZE) < (cyp->urb_r1->transfer_buffer_length + cyp->urb_r2->transfer_buffer_length))
{
printk(KERN_NOTICE "%s Bulk in Circular Buffer full", DRIVER_NAME);
goto exit;
}
/* resubmit urb if enough space is available in circular buffer */
if ((retval = usb_submit_urb(urb, GFP_KERNEL))) {
printk(KERN_ERR "%s Failed to resubmit read URB, error %d", DRIVER_NAME, retval);
goto exit;
}
usb_anchor_urb(urb, &cyp->read_urbs);
cyp->submitted_urb_r2 = true;
exit:
spin_unlock_irqrestore(&lock, flags);
return;
}
/* callback to open system call */
static int cypress_open(struct inode *inode, struct file *file)
{
int minor;
struct usb_interface *interface;
struct cypress_usb_dev *cyp;
unsigned int bulk_in_pipe;
int retval;
/* get minor number for device file */
minor = iminor(inode);
/* get USB interface for device file */
interface = usb_find_interface(&cypress_driver, minor);
if(!interface)
{
printk(KERN_ERR "%s Failed to find USB device", DRIVER_NAME);
return -ENODEV;
}
/* get device data */
cyp = usb_get_intfdata(interface);
if (!cyp) {
printk(KERN_ERR "%s No device data avialable in the USB device", DRIVER_NAME);
return -ENODEV;
}
/* increment our usage count for the device */
kref_get(&cyp->kref);
/* save object in the file's private structure */
file->private_data = cyp;
cyp->bulk_in_ring.buf = kzalloc(RING_BUFFER_SIZE, GFP_KERNEL);
if(cyp->bulk_in_ring.buf == NULL)
{
printk(KERN_ALERT "%s Failed to allocate memory for bulk input ring buffer.", DRIVER_NAME);
retval = -ENOMEM;
goto error_free_ring;
}
/* setup urbs for reading from device */
/* setup two urbs to ensure, that there is always a urb in the endpoints queue */
cyp->urb_r1 = usb_alloc_urb(0, GFP_KERNEL);
cyp->urb_r2 = usb_alloc_urb(0, GFP_KERNEL);
if (!cyp->urb_r1 || !cyp->urb_r2) {
printk(KERN_ALERT "%s Failed to allocate memory for URBs", DRIVER_NAME);
retval = -ENOMEM;
goto error_free_urb;
}
/* allocate dma-consistent buffer */
cyp->buffer_urb_in1 = usb_alloc_coherent(cyp->device, cyp->bulk_in_size, GFP_KERNEL, &cyp->dma_buffer_urb_in1);
cyp->buffer_urb_in2 = usb_alloc_coherent(cyp->device, cyp->bulk_in_size, GFP_KERNEL, &cyp->dma_buffer_urb_in2);
if (!cyp->buffer_urb_in1 || !cyp->buffer_urb_in2) {
printk(KERN_ALERT "%s Failed to allocate memory for read URBs data buffer", DRIVER_NAME);
retval = -ENOMEM;
goto error_free_dma;
}
/* create pipe for bulk input*/
bulk_in_pipe = usb_rcvbulkpipe(cyp->device, cyp->bulk_in_endpointAddr);
/* initialize the receiving urbs */
usb_fill_bulk_urb(cyp->urb_r1, cyp->device, bulk_in_pipe, cyp->buffer_urb_in1,\
cyp->bulk_in_size, cypress_read_bulk_callback1, cyp);
cyp->urb_r1->transfer_dma = cyp->dma_buffer_urb_in1;
cyp->urb_r1->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
usb_fill_bulk_urb(cyp->urb_r2, cyp->device, bulk_in_pipe, cyp->buffer_urb_in2,\
cyp->bulk_in_size, cypress_read_bulk_callback2, cyp);
cyp->urb_r2->transfer_dma = cyp->dma_buffer_urb_in2;
cyp->urb_r2->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
/* check if enough space is available in circular buffer */
if(CIRC_SPACE(cyp->bulk_in_ring.head, cyp->bulk_in_ring.tail, RING_BUFFER_SIZE) < (cyp->urb_r1->transfer_buffer_length + cyp->urb_r1->transfer_buffer_length))
{
printk(KERN_ERR "%s Bulk in Circular Buffer too small", DRIVER_NAME);
retval = -ENOMEM;
goto error_free_dma;
}
/* submit urb */
if ((retval = usb_submit_urb(cyp->urb_r1, GFP_KERNEL)) || (retval = usb_submit_urb(cyp->urb_r2, GFP_KERNEL))) {
printk(KERN_ERR "%s Failed to submit read URBs, error %d", DRIVER_NAME, retval);
goto error_kill_urb;
}
/* anchor urbs */
usb_anchor_urb(cyp->urb_r1, &cyp->read_urbs);
usb_anchor_urb(cyp->urb_r2, &cyp->read_urbs);
cyp->submitted_urb_r1 = true;
cyp->submitted_urb_r2 = true;
return 0;
/* handle exceptions */
error_kill_urb:
usb_kill_urb(cyp->urb_r1);
usb_kill_urb(cyp->urb_r2);
error_free_dma:
usb_free_coherent(cyp->device, cyp->bulk_in_size, cyp->buffer_urb_in1, cyp->dma_buffer_urb_in1);
usb_free_coherent(cyp->device, cyp->bulk_in_size, cyp->buffer_urb_in2, cyp->dma_buffer_urb_in2);
error_free_urb:
usb_free_urb(cyp->urb_r1);
usb_free_urb(cyp->urb_r2);
error_free_ring:
kfree(cyp->bulk_in_ring.buf);
return retval;
}
/* callback function for release system call */
static int cypress_release(struct inode *inode, struct file *file)
{
struct cypress_usb_dev *cyp;
cyp = file->private_data;
if(cyp == NULL) return -ENODEV;
/* decrement the count on our device */
kref_put(&cyp->kref, cypress_delete);
return 0;
}
/* callback function for write urb */
static void cypress_write_bulk_callback(struct urb *urb)
{
struct cypress_usb_dev *cyp;
/* get pointer to device struct */
cyp = urb->context;
printk(KERN_DEBUG "%s URB write callback triggered", DRIVER_NAME);
/* check urb status */
if (urb->status && !(urb->status == -ENOENT || urb->status == -ECONNRESET || urb->status == -ESHUTDOWN))
{
printk(KERN_ERR "%s Failed bulk write, error: %d", DRIVER_NAME, urb->status);
goto exit;
}
printk(KERN_DEBUG "%s URB write was successful", DRIVER_NAME);
exit:
/* free allocated buffer */
usb_free_coherent(urb->dev, urb->transfer_buffer_length, urb->transfer_buffer, urb->transfer_dma);
return;
/* don't need to free urb, this is done automatically */
}
/* callback to write system call */
static ssize_t cypress_write(struct file *file, const char __user *buffer, size_t count, loff_t *offset)
{
struct urb *urb;
struct cypress_usb_dev *cyp;
unsigned int bulk_out_pipe;
int retval;
/* verify that there is something to do */
if(count == 0) goto exit;
/* get pointer to device struct */
cyp = file->private_data;
/* allocate memory for urb */
urb = usb_alloc_urb(0, GFP_KERNEL);
if (!urb) {
printk(KERN_ALERT "%s Failed to allocate memory for URBs", DRIVER_NAME);
retval = -ENOMEM;
goto error;
}
/* allocate dma-consistent buffer */
cyp->bulk_out_buffer = usb_alloc_coherent(cyp->device, count, GFP_KERNEL, &urb->transfer_dma);
if (!cyp->bulk_out_buffer)
{
printk(KERN_ALERT "%s Failed to allocate memory for write URBs data buffer", DRIVER_NAME);
retval = -ENOMEM;
goto error;
}
/* copy data from user space */
if (copy_from_user(cyp->bulk_out_buffer, buffer, count))
{
printk(KERN_ERR "%s Failed to copy data from user space", DRIVER_NAME);
retval = -EFAULT;
goto error;
}
/* create pipe for bulk out */
bulk_out_pipe = usb_sndbulkpipe(cyp->device, cyp->bulk_out_endpointAddr);
/* initialize the urb */
usb_fill_bulk_urb(urb, cyp->device, bulk_out_pipe, cyp->bulk_out_buffer, \
count, cypress_write_bulk_callback, cyp);
urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
/* submit the urb */
retval = usb_submit_urb(urb, GFP_KERNEL);
if(retval)
{
printk(KERN_ERR "%s Failed to submit write URB, error %d", DRIVER_NAME, retval);
goto error;
}
usb_anchor_urb(urb, &cyp->write_urbs);
return count;
error:
usb_kill_urb(urb);
usb_free_coherent(cyp->device, count, cyp->bulk_out_buffer, urb->transfer_dma);
usb_free_urb(urb);
return retval;
exit:
return 0;
}
/* callback to read system call */
static ssize_t cypress_read(struct file *file, char *buffer, size_t count, loff_t *offset)
{
struct cypress_usb_dev *cyp;
int retval, circ_cnt, wrote_cnt;
unsigned long flags;
/* get pointer to device struct */
cyp = file->private_data;
if(cyp == NULL) return -ENODEV;
/* using spin_lock_irqsave - just because it is always save */
spin_lock_irqsave(&lock, flags);
/* read count available in buffer */
circ_cnt = CIRC_CNT(cyp->bulk_in_ring.head, cyp->bulk_in_ring.tail, RING_BUFFER_SIZE);
wrote_cnt = MIN(circ_cnt, count);
printk(KERN_NOTICE "%s Count is %d", DRIVER_NAME, wrote_cnt);
if(wrote_cnt == 0)
{
*buffer = '\0';
goto exit;
}
/* copy data to user space */
if(copy_to_user((void *)buffer, &cyp->bulk_in_ring.buf[cyp->bulk_in_ring.tail], wrote_cnt))
{
printk(KERN_ERR "%s Failed to copy data to user space", DRIVER_NAME);
retval = -EFAULT;
goto error;
}
/* update circular buffer index */
cyp->bulk_in_ring.tail = (cyp->bulk_in_ring.tail + wrote_cnt) & (RING_BUFFER_SIZE - 1);
printk(KERN_NOTICE "%s Tail at %d", DRIVER_NAME, cyp->bulk_in_ring.tail);
/* resubmit urb1 if necessary */
if(!cyp->submitted_urb_r1)
{
/* check if enough space is available in circular buffer */
if(CIRC_SPACE(cyp->bulk_in_ring.head, cyp->bulk_in_ring.tail, RING_BUFFER_SIZE) >= (cyp->urb_r1->transfer_buffer_length + cyp->urb_r2->transfer_buffer_length))
{
if ((retval = usb_submit_urb(cyp->urb_r1, GFP_KERNEL))) {
printk(KERN_ERR "%s Failed to resubmit read URB, error %d", DRIVER_NAME, retval);
goto error;
}
cyp->submitted_urb_r1 = true;
usb_anchor_urb(cyp->urb_r1, &cyp->read_urbs);
}
}
/* resubmit urb2 if necessary */
if(!cyp->submitted_urb_r2)
{
/* check if enough space is available in circular buffer */
if(CIRC_SPACE(cyp->bulk_in_ring.head, cyp->bulk_in_ring.tail, RING_BUFFER_SIZE) >= (cyp->urb_r1->transfer_buffer_length + cyp->urb_r2->transfer_buffer_length))
{
if ((retval = usb_submit_urb(cyp->urb_r2, GFP_KERNEL))) {
printk(KERN_ERR "%s Failed to resubmit read URB, error %d", DRIVER_NAME, retval);
goto error;
}
cyp->submitted_urb_r2 = true;
usb_anchor_urb(cyp->urb_r2, &cyp->read_urbs);
}
}
exit:
spin_unlock_irqrestore(&lock, flags);
return wrote_cnt;
error:
spin_unlock_irqrestore(&lock, flags);
return retval;
}
/* callback to open system call */
/* Returing 0 will sleep the kernel until an event happens in the queue */
static unsigned int cypress_poll(struct file *file, struct poll_table_struct *wait)
{
struct cypress_usb_dev *cyp;
int circ_cnt;
unsigned long flags, retval;
/* get pointer to device struct */
cyp = file->private_data;
if(cyp == NULL) return POLLERR;
/* this call does not sleep the kernel - it just makes the kernel call poll again if we return 0 */
poll_wait(file, &read_waitqueue, wait);
/* using spin_lock_irqsave - just because it is always save */
spin_lock_irqsave(&lock, flags);
/* read count available in buffer */
circ_cnt = CIRC_CNT(cyp->bulk_in_ring.head, cyp->bulk_in_ring.tail, RING_BUFFER_SIZE);
if(circ_cnt == 0)
{
/* no data to be read from circular buffer */
retval = 0; /* return 0 sleeps the kernel */
goto exit;
}
else
{
retval = POLLIN | POLLRDNORM;
goto exit;
}
exit:
spin_unlock_irqrestore(&lock, flags);
return retval;
}
/* disconnect is called if device is removed */
static void cypress_disconnect(struct usb_interface *interface)
{
struct cypress_usb_dev *cyp;
/* get USB interface for device file */
cyp = usb_get_intfdata(interface);
/* remove information from interface */
usb_set_intfdata(interface, NULL);
/* release minor */
usb_deregister_dev(interface, &cypress_class);
/* decrement our usage count */
kref_put(&cyp->kref, cypress_delete);
return;
}
/* register the usb driver with the USB subsystem */
static int __init usb_cypress_init(void)
{
int result;
printk(KERN_INFO "%s [*] Register Cypress USB Bulk Transfer driver with the USB subsystem", DRIVER_NAME);
/* register this driver with the USB subsystem */
result = usb_register(&cypress_driver);
if(result) {
printk(KERN_ALERT "%s\t Failed to register Cypress USB Bulk Transfer driver with the USB subsystem, error %d", DRIVER_NAME, result);
usb_deregister(&cypress_driver);
return result;
}
return 0;
}
module_init(usb_cypress_init);
/* deregister the usb driver with the USB subsystem */
static void __exit usb_cypress_exit(void)
{
printk(KERN_INFO "%s [*] Deregistered Cypress USB Bulk Transfer driver with the USB subsystem", DRIVER_NAME);
/* deregister this driver with the USB subsystem */
usb_deregister(&cypress_driver);
}
module_exit(usb_cypress_exit);
내 사용자 공간 애플리케이션은 다음과 같습니다.
static void read_thread_cleanup_handler(void *arg)
{
log_info("Terminate read thread - cleanup handler");
}
void *read_thread_function(void *conf_values)
{
FILE *dest_file;
struct pollfd poll_fd;
char *buffer;
int usb_dev, ret;
ssize_t read_count;
configValues_typ *config_values = (configValues_typ *)conf_values;
/* push thread cleanup handler to stack */
pthread_cleanup_push(read_thread_cleanup_handler, NULL);
log_info("Read Thread started: %d", getpid());
/* fill poll struct */
poll_fd.fd = config_values->usb_dev;
poll_fd.events = POLLIN;
dest_file = fopen(config_values->dest_file_path, "a+");
if(dest_file == NULL)
{
log_error("Failed to open destination file %s, error: %s", config_values->dest_file_path, strerror(errno));
ret = -2;
goto read_thread_error;
}
buffer = malloc(BUFFER_SIZE);
if(buffer == NULL)
{
log_error("Failed to allocate memory for buffer, error: %s",strerror(errno));
ret = -3;
goto read_thread_error;
}
while(!terminate)
{
/* poll to read file (blocking) - using an infinite timeout */
if(poll(&poll_fd, 1, -1) < 0)
{
log_error("Failed polling Cypress device driver for data to read, error: %s", strerror(errno));
ret = -4;
goto read_thread_error;
}
read_count = read(config_values->usb_dev, buffer, BUFFER_SIZE);
if(read_count < 0)
{
log_error("Failed to read from Cypress device driver, error: %s", strerror(errno));
ret = -5;
goto read_thread_error;
}
else if(read_count > 0)
{
if(fprintf(dest_file, "%s",buffer) < 0)
{
log_error("Failed to write to destination file %s, error: %s", config_values->dest_file_path, strerror(errno));
ret = -6;
goto read_thread_error;
}
}
}
pthread_cleanup_pop(1);
return NULL;
read_thread_error:
log_error("Terminate read thread on error");
pthread_exit(&ret);
}