⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 relay.c

📁 Kernel code of linux kernel
💻 C
📖 第 1 页 / 共 3 页
字号:
/* * Public API and common code for kernel->userspace relay file support. * * See Documentation/filesystems/relay.txt for an overview. * * Copyright (C) 2002-2005 - Tom Zanussi (zanussi@us.ibm.com), IBM Corp * Copyright (C) 1999-2005 - Karim Yaghmour (karim@opersys.com) * * Moved to kernel/relay.c by Paul Mundt, 2006. * November 2006 - CPU hotplug support by Mathieu Desnoyers * 	(mathieu.desnoyers@polymtl.ca) * * This file is released under the GPL. */#include <linux/errno.h>#include <linux/stddef.h>#include <linux/slab.h>#include <linux/module.h>#include <linux/string.h>#include <linux/relay.h>#include <linux/vmalloc.h>#include <linux/mm.h>#include <linux/cpu.h>#include <linux/splice.h>/* list of open channels, for cpu hotplug */static DEFINE_MUTEX(relay_channels_mutex);static LIST_HEAD(relay_channels);/* * close() vm_op implementation for relay file mapping. */static void relay_file_mmap_close(struct vm_area_struct *vma){	struct rchan_buf *buf = vma->vm_private_data;	buf->chan->cb->buf_unmapped(buf, vma->vm_file);}/* * fault() vm_op implementation for relay file mapping. */static int relay_buf_fault(struct vm_area_struct *vma, struct vm_fault *vmf){	struct page *page;	struct rchan_buf *buf = vma->vm_private_data;	pgoff_t pgoff = vmf->pgoff;	if (!buf)		return VM_FAULT_OOM;	page = vmalloc_to_page(buf->start + (pgoff << PAGE_SHIFT));	if (!page)		return VM_FAULT_SIGBUS;	get_page(page);	vmf->page = page;	return 0;}/* * vm_ops for relay file mappings. */static struct vm_operations_struct relay_file_mmap_ops = {	.fault = relay_buf_fault,	.close = relay_file_mmap_close,};/* * allocate an array of pointers of struct page */static struct page **relay_alloc_page_array(unsigned int n_pages){	struct page **array;	size_t pa_size = n_pages * sizeof(struct page *);	if (pa_size > PAGE_SIZE) {		array = vmalloc(pa_size);		if (array)			memset(array, 0, pa_size);	} else {		array = kzalloc(pa_size, GFP_KERNEL);	}	return array;}/* * free an array of pointers of struct page */static void relay_free_page_array(struct page **array){	if (is_vmalloc_addr(array))		vfree(array);	else		kfree(array);}/** *	relay_mmap_buf: - mmap channel buffer to process address space *	@buf: relay channel buffer *	@vma: vm_area_struct describing memory to be mapped * *	Returns 0 if ok, negative on error * *	Caller should already have grabbed mmap_sem. */static int relay_mmap_buf(struct rchan_buf *buf, struct vm_area_struct *vma){	unsigned long length = vma->vm_end - vma->vm_start;	struct file *filp = vma->vm_file;	if (!buf)		return -EBADF;	if (length != (unsigned long)buf->chan->alloc_size)		return -EINVAL;	vma->vm_ops = &relay_file_mmap_ops;	vma->vm_flags |= VM_DONTEXPAND;	vma->vm_private_data = buf;	buf->chan->cb->buf_mapped(buf, filp);	return 0;}/** *	relay_alloc_buf - allocate a channel buffer *	@buf: the buffer struct *	@size: total size of the buffer * *	Returns a pointer to the resulting buffer, %NULL if unsuccessful. The *	passed in size will get page aligned, if it isn't already. */static void *relay_alloc_buf(struct rchan_buf *buf, size_t *size){	void *mem;	unsigned int i, j, n_pages;	*size = PAGE_ALIGN(*size);	n_pages = *size >> PAGE_SHIFT;	buf->page_array = relay_alloc_page_array(n_pages);	if (!buf->page_array)		return NULL;	for (i = 0; i < n_pages; i++) {		buf->page_array[i] = alloc_page(GFP_KERNEL);		if (unlikely(!buf->page_array[i]))			goto depopulate;		set_page_private(buf->page_array[i], (unsigned long)buf);	}	mem = vmap(buf->page_array, n_pages, VM_MAP, PAGE_KERNEL);	if (!mem)		goto depopulate;	memset(mem, 0, *size);	buf->page_count = n_pages;	return mem;depopulate:	for (j = 0; j < i; j++)		__free_page(buf->page_array[j]);	relay_free_page_array(buf->page_array);	return NULL;}/** *	relay_create_buf - allocate and initialize a channel buffer *	@chan: the relay channel * *	Returns channel buffer if successful, %NULL otherwise. */static struct rchan_buf *relay_create_buf(struct rchan *chan){	struct rchan_buf *buf = kzalloc(sizeof(struct rchan_buf), GFP_KERNEL);	if (!buf)		return NULL;	buf->padding = kmalloc(chan->n_subbufs * sizeof(size_t *), GFP_KERNEL);	if (!buf->padding)		goto free_buf;	buf->start = relay_alloc_buf(buf, &chan->alloc_size);	if (!buf->start)		goto free_buf;	buf->chan = chan;	kref_get(&buf->chan->kref);	return buf;free_buf:	kfree(buf->padding);	kfree(buf);	return NULL;}/** *	relay_destroy_channel - free the channel struct *	@kref: target kernel reference that contains the relay channel * *	Should only be called from kref_put(). */static void relay_destroy_channel(struct kref *kref){	struct rchan *chan = container_of(kref, struct rchan, kref);	kfree(chan);}/** *	relay_destroy_buf - destroy an rchan_buf struct and associated buffer *	@buf: the buffer struct */static void relay_destroy_buf(struct rchan_buf *buf){	struct rchan *chan = buf->chan;	unsigned int i;	if (likely(buf->start)) {		vunmap(buf->start);		for (i = 0; i < buf->page_count; i++)			__free_page(buf->page_array[i]);		relay_free_page_array(buf->page_array);	}	chan->buf[buf->cpu] = NULL;	kfree(buf->padding);	kfree(buf);	kref_put(&chan->kref, relay_destroy_channel);}/** *	relay_remove_buf - remove a channel buffer *	@kref: target kernel reference that contains the relay buffer * *	Removes the file from the fileystem, which also frees the *	rchan_buf_struct and the channel buffer.  Should only be called from *	kref_put(). */static void relay_remove_buf(struct kref *kref){	struct rchan_buf *buf = container_of(kref, struct rchan_buf, kref);	buf->chan->cb->remove_buf_file(buf->dentry);	relay_destroy_buf(buf);}/** *	relay_buf_empty - boolean, is the channel buffer empty? *	@buf: channel buffer * *	Returns 1 if the buffer is empty, 0 otherwise. */static int relay_buf_empty(struct rchan_buf *buf){	return (buf->subbufs_produced - buf->subbufs_consumed) ? 0 : 1;}/** *	relay_buf_full - boolean, is the channel buffer full? *	@buf: channel buffer * *	Returns 1 if the buffer is full, 0 otherwise. */int relay_buf_full(struct rchan_buf *buf){	size_t ready = buf->subbufs_produced - buf->subbufs_consumed;	return (ready >= buf->chan->n_subbufs) ? 1 : 0;}EXPORT_SYMBOL_GPL(relay_buf_full);/* * High-level relay kernel API and associated functions. *//* * rchan_callback implementations defining default channel behavior.  Used * in place of corresponding NULL values in client callback struct. *//* * subbuf_start() default callback.  Does nothing. */static int subbuf_start_default_callback (struct rchan_buf *buf,					  void *subbuf,					  void *prev_subbuf,					  size_t prev_padding){	if (relay_buf_full(buf))		return 0;	return 1;}/* * buf_mapped() default callback.  Does nothing. */static void buf_mapped_default_callback(struct rchan_buf *buf,					struct file *filp){}/* * buf_unmapped() default callback.  Does nothing. */static void buf_unmapped_default_callback(struct rchan_buf *buf,					  struct file *filp){}/* * create_buf_file_create() default callback.  Does nothing. */static struct dentry *create_buf_file_default_callback(const char *filename,						       struct dentry *parent,						       int mode,						       struct rchan_buf *buf,						       int *is_global){	return NULL;}/* * remove_buf_file() default callback.  Does nothing. */static int remove_buf_file_default_callback(struct dentry *dentry){	return -EINVAL;}/* relay channel default callbacks */static struct rchan_callbacks default_channel_callbacks = {	.subbuf_start = subbuf_start_default_callback,	.buf_mapped = buf_mapped_default_callback,	.buf_unmapped = buf_unmapped_default_callback,	.create_buf_file = create_buf_file_default_callback,	.remove_buf_file = remove_buf_file_default_callback,};/** *	wakeup_readers - wake up readers waiting on a channel *	@data: contains the channel buffer * *	This is the timer function used to defer reader waking. */static void wakeup_readers(unsigned long data){	struct rchan_buf *buf = (struct rchan_buf *)data;	wake_up_interruptible(&buf->read_wait);}/** *	__relay_reset - reset a channel buffer *	@buf: the channel buffer *	@init: 1 if this is a first-time initialization * *	See relay_reset() for description of effect. */static void __relay_reset(struct rchan_buf *buf, unsigned int init){	size_t i;	if (init) {		init_waitqueue_head(&buf->read_wait);		kref_init(&buf->kref);		setup_timer(&buf->timer, wakeup_readers, (unsigned long)buf);	} else		del_timer_sync(&buf->timer);	buf->subbufs_produced = 0;	buf->subbufs_consumed = 0;	buf->bytes_consumed = 0;	buf->finalized = 0;	buf->data = buf->start;	buf->offset = 0;	for (i = 0; i < buf->chan->n_subbufs; i++)		buf->padding[i] = 0;	buf->chan->cb->subbuf_start(buf, buf->data, NULL, 0);}/** *	relay_reset - reset the channel *	@chan: the channel * *	This has the effect of erasing all data from all channel buffers *	and restarting the channel in its initial state.  The buffers *	are not freed, so any mappings are still in effect. * *	NOTE. Care should be taken that the channel isn't actually *	being used by anything when this call is made. */void relay_reset(struct rchan *chan){	unsigned int i;	if (!chan)		return;	if (chan->is_global && chan->buf[0]) {		__relay_reset(chan->buf[0], 0);		return;	}	mutex_lock(&relay_channels_mutex);	for_each_online_cpu(i)		if (chan->buf[i])			__relay_reset(chan->buf[i], 0);	mutex_unlock(&relay_channels_mutex);}EXPORT_SYMBOL_GPL(relay_reset);static inline void relay_set_buf_dentry(struct rchan_buf *buf,					struct dentry *dentry){	buf->dentry = dentry;	buf->dentry->d_inode->i_size = buf->early_bytes;}static struct dentry *relay_create_buf_file(struct rchan *chan,					    struct rchan_buf *buf,					    unsigned int cpu){	struct dentry *dentry;	char *tmpname;	tmpname = kzalloc(NAME_MAX + 1, GFP_KERNEL);	if (!tmpname)		return NULL;	snprintf(tmpname, NAME_MAX, "%s%d", chan->base_filename, cpu);	/* Create file in fs */	dentry = chan->cb->create_buf_file(tmpname, chan->parent,					   S_IRUSR, buf,					   &chan->is_global);	kfree(tmpname);	return dentry;}/* *	relay_open_buf - create a new relay channel buffer * *	used by relay_open() and CPU hotplug. */static struct rchan_buf *relay_open_buf(struct rchan *chan, unsigned int cpu){ 	struct rchan_buf *buf = NULL;	struct dentry *dentry; 	if (chan->is_global)		return chan->buf[0];	buf = relay_create_buf(chan);	if (!buf)		return NULL;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -