Commit 177a429c authored by Dave Jones's avatar Dave Jones Committed by Linus Torvalds

[PATCH] proc_file_read documentation/buffer overflow detection.

This one has been around for ages, has Rusty's name
all over it, but wasn't from him.
48 lines of comments, and no swearing? 8-)

Mostly documentation, but also has some buffer overflow
detection in there.
parent 1b30301d
...@@ -65,43 +65,108 @@ proc_file_read(struct file * file, char * buf, size_t nbytes, loff_t *ppos) ...@@ -65,43 +65,108 @@ proc_file_read(struct file * file, char * buf, size_t nbytes, loff_t *ppos)
start = NULL; start = NULL;
if (dp->get_info) { if (dp->get_info) {
/* /* Handle old net routines */
* Handle backwards compatibility with the old net
* routines.
*/
n = dp->get_info(page, &start, *ppos, count); n = dp->get_info(page, &start, *ppos, count);
if (n < count) if (n < count)
eof = 1; eof = 1;
} else if (dp->read_proc) { } else if (dp->read_proc) {
/*
* How to be a proc read function
* ------------------------------
* Prototype:
* int f(char *buffer, char **start, off_t offset,
* int count, int *peof, void *dat)
*
* Assume that the buffer is "count" bytes in size.
*
* If you know you have supplied all the data you
* have, set *peof.
*
* You have three ways to return data:
* 0) Leave *start = NULL. (This is the default.)
* Put the data of the requested offset at that
* offset within the buffer. Return the number (n)
* of bytes there are from the beginning of the
* buffer up to the last byte of data. If the
* number of supplied bytes (= n - offset) is
* greater than zero and you didn't signal eof
* and the reader is prepared to take more data
* you will be called again with the requested
* offset advanced by the number of bytes
* absorbed. This interface is useful for files
* no larger than the buffer.
* 1) Set *start = an unsigned long value less than
* the buffer address but greater than zero.
* Put the data of the requested offset at the
* beginning of the buffer. Return the number of
* bytes of data placed there. If this number is
* greater than zero and you didn't signal eof
* and the reader is prepared to take more data
* you will be called again with the requested
* offset advanced by *start. This interface is
* useful when you have a large file consisting
* of a series of blocks which you want to count
* and return as wholes.
* (Hack by Paul.Russell@rustcorp.com.au)
* 2) Set *start = an address within the buffer.
* Put the data of the requested offset at *start.
* Return the number of bytes of data placed there.
* If this number is greater than zero and you
* didn't signal eof and the reader is prepared to
* take more data you will be called again with the
* requested offset advanced by the number of bytes
* absorbed.
*/
n = dp->read_proc(page, &start, *ppos, n = dp->read_proc(page, &start, *ppos,
count, &eof, dp->data); count, &eof, dp->data);
} else } else
break; break;
if (!start) { if (n == 0) /* end of file */
/*
* For proc files that are less than 4k
*/
start = page + *ppos;
n -= *ppos;
if (n <= 0)
break; break;
if (n > count) if (n < 0) { /* error */
n = count;
}
if (n == 0)
break; /* End of file */
if (n < 0) {
if (retval == 0) if (retval == 0)
retval = n; retval = n;
break; break;
} }
/* This is a hack to allow mangling of file pos independent if (start == NULL) {
* of actual bytes read. Simply place the data at page, if (n > PAGE_SIZE) {
* return the bytes, and set `start' to the desired offset printk(KERN_ERR
* as an unsigned int. - Paul.Russell@rustcorp.com.au "proc_file_read: Apparent buffer overflow!\n");
n = PAGE_SIZE;
}
if (n > count)
n = count;
n -= *ppos;
if (n <= 0)
break;
start = page + *ppos;
} else if (start < page) {
if (n > PAGE_SIZE) {
printk(KERN_ERR
"proc_file_read: Apparent buffer overflow!\n");
n = PAGE_SIZE;
}
if (n > count) {
/*
* Don't reduce n because doing so might
* cut off part of a data block.
*/ */
printk(KERN_WARNING
"proc_file_read: Read count exceeded\n");
}
} else /* start >= page */ {
unsigned long startoff = (unsigned long)(start - page);
if (n > (PAGE_SIZE - startoff)) {
printk(KERN_ERR
"proc_file_read: Apparent buffer overflow!\n");
n = PAGE_SIZE - startoff;
}
if (n > count)
n = count;
}
n -= copy_to_user(buf, start < page ? page : start, n); n -= copy_to_user(buf, start < page ? page : start, n);
if (n == 0) { if (n == 0) {
if (retval == 0) if (retval == 0)
...@@ -109,7 +174,7 @@ proc_file_read(struct file * file, char * buf, size_t nbytes, loff_t *ppos) ...@@ -109,7 +174,7 @@ proc_file_read(struct file * file, char * buf, size_t nbytes, loff_t *ppos)
break; break;
} }
*ppos += start < page ? (long)start : n; /* Move down the file */ *ppos += start < page ? (unsigned long)start : n;
nbytes -= n; nbytes -= n;
buf += n; buf += n;
retval += n; retval += n;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment