fw-cdev.c
来自「linux 内核源代码」· C语言 代码 · 共 1,005 行 · 第 1/2 页
C
1,005 行
/* * Char device for device raw access * * Copyright (C) 2005-2007 Kristian Hoegsberg <krh@bitplanet.net> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */#include <linux/module.h>#include <linux/kernel.h>#include <linux/wait.h>#include <linux/errno.h>#include <linux/device.h>#include <linux/vmalloc.h>#include <linux/poll.h>#include <linux/preempt.h>#include <linux/time.h>#include <linux/delay.h>#include <linux/mm.h>#include <linux/idr.h>#include <linux/compat.h>#include <linux/firewire-cdev.h>#include <asm/system.h>#include <asm/uaccess.h>#include "fw-transaction.h"#include "fw-topology.h"#include "fw-device.h"struct client;struct client_resource { struct list_head link; void (*release)(struct client *client, struct client_resource *r); u32 handle;};/* * dequeue_event() just kfree()'s the event, so the event has to be * the first field in the struct. */struct event { struct { void *data; size_t size; } v[2]; struct list_head link;};struct bus_reset { struct event event; struct fw_cdev_event_bus_reset reset;};struct response { struct event event; struct fw_transaction transaction; struct client *client; struct client_resource resource; struct fw_cdev_event_response response;};struct iso_interrupt { struct event event; struct fw_cdev_event_iso_interrupt interrupt;};struct client { u32 version; struct fw_device *device; spinlock_t lock; u32 resource_handle; struct list_head resource_list; struct list_head event_list; wait_queue_head_t wait; u64 bus_reset_closure; struct fw_iso_context *iso_context; u64 iso_closure; struct fw_iso_buffer buffer; unsigned long vm_start; struct list_head link;};static inline void __user *u64_to_uptr(__u64 value){ return (void __user *)(unsigned long)value;}static inline __u64uptr_to_u64(void __user *ptr){ return (__u64)(unsigned long)ptr;}static int fw_device_op_open(struct inode *inode, struct file *file){ struct fw_device *device; struct client *client; unsigned long flags; device = fw_device_from_devt(inode->i_rdev); if (device == NULL) return -ENODEV; client = kzalloc(sizeof(*client), GFP_KERNEL); if (client == NULL) return -ENOMEM; client->device = fw_device_get(device); INIT_LIST_HEAD(&client->event_list); INIT_LIST_HEAD(&client->resource_list); spin_lock_init(&client->lock); init_waitqueue_head(&client->wait); file->private_data = client; spin_lock_irqsave(&device->card->lock, flags); list_add_tail(&client->link, &device->client_list); spin_unlock_irqrestore(&device->card->lock, flags); return 0;}static void queue_event(struct client *client, struct event *event, void *data0, size_t size0, void *data1, size_t size1){ unsigned long flags; event->v[0].data = data0; event->v[0].size = size0; event->v[1].data = data1; event->v[1].size = size1; spin_lock_irqsave(&client->lock, flags); list_add_tail(&event->link, &client->event_list); spin_unlock_irqrestore(&client->lock, flags); wake_up_interruptible(&client->wait);}static intdequeue_event(struct client *client, char __user *buffer, size_t count){ unsigned long flags; struct event *event; size_t size, total; int i, retval; retval = wait_event_interruptible(client->wait, !list_empty(&client->event_list) || fw_device_is_shutdown(client->device)); if (retval < 0) return retval; if (list_empty(&client->event_list) && fw_device_is_shutdown(client->device)) return -ENODEV; spin_lock_irqsave(&client->lock, flags); event = container_of(client->event_list.next, struct event, link); list_del(&event->link); spin_unlock_irqrestore(&client->lock, flags); total = 0; for (i = 0; i < ARRAY_SIZE(event->v) && total < count; i++) { size = min(event->v[i].size, count - total); if (copy_to_user(buffer + total, event->v[i].data, size)) { retval = -EFAULT; goto out; } total += size; } retval = total; out: kfree(event); return retval;}static ssize_tfw_device_op_read(struct file *file, char __user *buffer, size_t count, loff_t *offset){ struct client *client = file->private_data; return dequeue_event(client, buffer, count);}static voidfill_bus_reset_event(struct fw_cdev_event_bus_reset *event, struct client *client){ struct fw_card *card = client->device->card; event->closure = client->bus_reset_closure; event->type = FW_CDEV_EVENT_BUS_RESET; event->node_id = client->device->node_id; event->local_node_id = card->local_node->node_id; event->bm_node_id = 0; /* FIXME: We don't track the BM. */ event->irm_node_id = card->irm_node->node_id; event->root_node_id = card->root_node->node_id; event->generation = card->generation;}static voidfor_each_client(struct fw_device *device, void (*callback)(struct client *client)){ struct fw_card *card = device->card; struct client *c; unsigned long flags; spin_lock_irqsave(&card->lock, flags); list_for_each_entry(c, &device->client_list, link) callback(c); spin_unlock_irqrestore(&card->lock, flags);}static voidqueue_bus_reset_event(struct client *client){ struct bus_reset *bus_reset; bus_reset = kzalloc(sizeof(*bus_reset), GFP_ATOMIC); if (bus_reset == NULL) { fw_notify("Out of memory when allocating bus reset event\n"); return; } fill_bus_reset_event(&bus_reset->reset, client); queue_event(client, &bus_reset->event, &bus_reset->reset, sizeof(bus_reset->reset), NULL, 0);}void fw_device_cdev_update(struct fw_device *device){ for_each_client(device, queue_bus_reset_event);}static void wake_up_client(struct client *client){ wake_up_interruptible(&client->wait);}void fw_device_cdev_remove(struct fw_device *device){ for_each_client(device, wake_up_client);}static int ioctl_get_info(struct client *client, void *buffer){ struct fw_cdev_get_info *get_info = buffer; struct fw_cdev_event_bus_reset bus_reset; client->version = get_info->version; get_info->version = FW_CDEV_VERSION; if (get_info->rom != 0) { void __user *uptr = u64_to_uptr(get_info->rom); size_t want = get_info->rom_length; size_t have = client->device->config_rom_length * 4; if (copy_to_user(uptr, client->device->config_rom, min(want, have))) return -EFAULT; } get_info->rom_length = client->device->config_rom_length * 4; client->bus_reset_closure = get_info->bus_reset_closure; if (get_info->bus_reset != 0) { void __user *uptr = u64_to_uptr(get_info->bus_reset); fill_bus_reset_event(&bus_reset, client); if (copy_to_user(uptr, &bus_reset, sizeof(bus_reset))) return -EFAULT; } get_info->card = client->device->card->index; return 0;}static voidadd_client_resource(struct client *client, struct client_resource *resource){ unsigned long flags; spin_lock_irqsave(&client->lock, flags); list_add_tail(&resource->link, &client->resource_list); resource->handle = client->resource_handle++; spin_unlock_irqrestore(&client->lock, flags);}static intrelease_client_resource(struct client *client, u32 handle, struct client_resource **resource){ struct client_resource *r; unsigned long flags; spin_lock_irqsave(&client->lock, flags); list_for_each_entry(r, &client->resource_list, link) { if (r->handle == handle) { list_del(&r->link); break; } } spin_unlock_irqrestore(&client->lock, flags); if (&r->link == &client->resource_list) return -EINVAL; if (resource) *resource = r; else r->release(client, r); return 0;}static voidrelease_transaction(struct client *client, struct client_resource *resource){ struct response *response = container_of(resource, struct response, resource); fw_cancel_transaction(client->device->card, &response->transaction);}static voidcomplete_transaction(struct fw_card *card, int rcode, void *payload, size_t length, void *data){ struct response *response = data; struct client *client = response->client; unsigned long flags; if (length < response->response.length) response->response.length = length; if (rcode == RCODE_COMPLETE) memcpy(response->response.data, payload, response->response.length); spin_lock_irqsave(&client->lock, flags); list_del(&response->resource.link); spin_unlock_irqrestore(&client->lock, flags); response->response.type = FW_CDEV_EVENT_RESPONSE; response->response.rcode = rcode; queue_event(client, &response->event, &response->response, sizeof(response->response), response->response.data, response->response.length);}static int ioctl_send_request(struct client *client, void *buffer){ struct fw_device *device = client->device; struct fw_cdev_send_request *request = buffer; struct response *response; /* What is the biggest size we'll accept, really? */ if (request->length > 4096) return -EINVAL; response = kmalloc(sizeof(*response) + request->length, GFP_KERNEL); if (response == NULL) return -ENOMEM; response->client = client; response->response.length = request->length; response->response.closure = request->closure; if (request->data && copy_from_user(response->response.data, u64_to_uptr(request->data), request->length)) { kfree(response); return -EFAULT; } response->resource.release = release_transaction; add_client_resource(client, &response->resource); fw_send_request(device->card, &response->transaction, request->tcode & 0x1f, device->node->node_id, request->generation, device->max_speed, request->offset, response->response.data, request->length, complete_transaction, response); if (request->data) return sizeof(request) + request->length; else return sizeof(request);}struct address_handler { struct fw_address_handler handler; __u64 closure; struct client *client; struct client_resource resource;};struct request { struct fw_request *request; void *data; size_t length; struct client_resource resource;};struct request_event { struct event event; struct fw_cdev_event_request request;};static voidrelease_request(struct client *client, struct client_resource *resource){ struct request *request = container_of(resource, struct request, resource); fw_send_response(client->device->card, request->request, RCODE_CONFLICT_ERROR); kfree(request);}static voidhandle_request(struct fw_card *card, struct fw_request *r, int tcode, int destination, int source, int generation, int speed, unsigned long long offset, void *payload, size_t length, void *callback_data){ struct address_handler *handler = callback_data; struct request *request; struct request_event *e; struct client *client = handler->client; request = kmalloc(sizeof(*request), GFP_ATOMIC); e = kmalloc(sizeof(*e), GFP_ATOMIC); if (request == NULL || e == NULL) { kfree(request); kfree(e); fw_send_response(card, r, RCODE_CONFLICT_ERROR); return; } request->request = r; request->data = payload; request->length = length; request->resource.release = release_request; add_client_resource(client, &request->resource); e->request.type = FW_CDEV_EVENT_REQUEST; e->request.tcode = tcode; e->request.offset = offset; e->request.length = length; e->request.handle = request->resource.handle; e->request.closure = handler->closure; queue_event(client, &e->event, &e->request, sizeof(e->request), payload, length);}static voidrelease_address_handler(struct client *client, struct client_resource *resource){ struct address_handler *handler = container_of(resource, struct address_handler, resource); fw_core_remove_address_handler(&handler->handler); kfree(handler);}static int ioctl_allocate(struct client *client, void *buffer){ struct fw_cdev_allocate *request = buffer; struct address_handler *handler; struct fw_address_region region; handler = kmalloc(sizeof(*handler), GFP_KERNEL); if (handler == NULL) return -ENOMEM; region.start = request->offset;
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?