Device Driver 관련 예제를 분석중입니다

windower의 이미지

간단하게 가상 메모리를 동적으로 할당하고
거기에 읽고 쓰는 메모리 Device Driver를 분석하고 있습니다.
간단히 쏘스는 아래와 같습니다.

#include <linux/kernel.h>
#include <linux/module.h>

#include <linux/fs.h>
#include <linux/kdev_t.h>
#include <asm/uaccess.h>

#define DEVICE_NAME "minibuf"
#define BUFFER_LEN 1024

// Global variables
static int s_nMajor = 0;
static int s_bDeviceOpen = 0;
static char s_strBuf[BUFFER_LEN];
static int s_nBufPos = 0, s_nBufEnd = 0;

// Function Prototypes
static int device_open(struct inode *inode, struct file *flip);
static int device_release(struct inode *inode, struct file *flip);
static ssize_t device_read(struct file *flip, char *buffer, size_t length,loff_t *offset);
static ssize_t device_write(struct file *flip, char *buffer, size_t length, loff_t *offset);
static int is_buffer_empty(void);
static int is_buffer_full(void);
static int read_buffer_char(char *buffer);
static int write_buffer_char(char *buffer);

static struct file_operations device_fops =
{
	read:device_read,     /* read */
	write:device_write,    /* write */
	open:device_open,     /* open */
	release:device_release   /* release */
};

int init_module()
{
	printk("RAM Buffer Module loading...\n");
	if( ( s_nMajor = register_chrdev(0, DEVICE_NAME, &device_fops) )<0 )
	{
		printk(DEVICE_NAME" : Device Registration Failed ~! (%d) \n",s_nMajor);
		return s_nMajor;
	}
	printk(DEVICE_NAME" : Device Registration with Major Number = %d\n",s_nMajor);
	strcpy(s_strBuf,"안녕하세요 세상들이셔~!\n");
	s_nBufEnd = strlen(s_strBuf) + 1;
	return 0;
}

void cleanup_module(void)
{
	int nRetCode;
	printk("Unloading RAM Buffer Module\n");
	if( (nRetCode = unregister_chrdev(s_nMajor, DEVICE_NAME))<0 )
	{
		printk(DEVICE_NAME" : Device unregisteration failed (%d)\n",nRetCode);
	}
}

int device_open(struct inode *inode, struct file *flip)
{
	printk(DEVICE_NAME" : Device open (%d, %d)\n",MAJOR(inode->i_rdev), MINOR(inode->i_rdev) );
	if(s_bDeviceOpen)
	{
		printk(DEVICE_NAME" : Device already open\n");
		return -EBUSY;
	}
	++s_bDeviceOpen;
	MOD_INC_USE_COUNT;
	return 0;
}

int device_release(struct inode *inode, struct file *flip)
{
	printk(DEVICE_NAME" : Device Release (%d, %d)\n",MAJOR(inode->i_rdev), MINOR(inode->i_rdev) );
	if(!s_bDeviceOpen)
	{
		printk(DEVICE_NAME" : Device has not opened\n");
		return -EINVAL;
	}
	--s_bDeviceOpen;
	MOD_DEC_USE_COUNT;
	return 0;									
}

ssize_t device_read(struct file *filp, char *buffer, size_t length, loff_t *offset)
{
	int count = 0;
	if (is_buffer_empty()) {
		printk(DEVICE_NAME " : Read return EOF\n");
		return 0;
	} // end if
	while (!is_buffer_empty() && length > 1) {
		read_buffer_char(buffer);
		++buffer;
		--length;
		++count;
	} // end while
	put_user(0, buffer);
	++count;
	printk(DEVICE_NAME " : read %d bytes\n", count);
	return count;
} // end func

ssize_t device_write(struct file *filp, char *buffer, size_t length, loff_t *offset)
{
	return -ENOSYS;
} // end func

// Buffer Management
int is_buffer_empty() {
	return (s_nBufPos == s_nBufEnd) ? 1 : 0;
} // end func

int is_buffer_full() {
	int pos = s_nBufEnd + 1;
	if (pos == BUFFER_LEN) pos = 0;
	return (pos == s_nBufPos) ? 1 : 0;
} // end func

int read_buffer_char(char *buffer) {
	if (is_buffer_empty()) return -1;
	put_user(s_strBuf[s_nBufPos], buffer);
	if (++s_nBufPos == BUFFER_LEN) s_nBufPos = 0;
	return 0;
} // end func

int write_buffer_char(char *buffer) {
	if (is_buffer_full()) return -1;
	get_user(s_strBuf[s_nBufEnd], buffer);
	if (++s_nBufEnd == BUFFER_LEN) s_nBufEnd = 0;
	return 0;
} // end func

다름이 아니라 여기에 나온 put_user와 get_user부분에 관련된
manual을 알고 싶은데 man put_user로 해서도 찾을 수 없었습니다.
관련된 부분을 어떻게 찾을 수 있을지요?

그리고 아마 여기서 가장 중요한
함수인듯한데, 간단히라도 좋으니 put_user와
get_user의 기능을 알려주신다면 감사하겠습니다.

charsyam의 이미지

windower wrote:
간단하게 가상 메모리를 동적으로 할당하고
거기에 읽고 쓰는 메모리 Device Driver를 분석하고 있습니다.
간단히 쏘스는 아래와 같습니다.

#include <linux/kernel.h>
#include <linux/module.h>

#include <linux/fs.h>
#include <linux/kdev_t.h>
#include <asm/uaccess.h>

#define DEVICE_NAME "minibuf"
#define BUFFER_LEN 1024

// Global variables
static int s_nMajor = 0;
static int s_bDeviceOpen = 0;
static char s_strBuf[BUFFER_LEN];
static int s_nBufPos = 0, s_nBufEnd = 0;

// Function Prototypes
static int device_open(struct inode *inode, struct file *flip);
static int device_release(struct inode *inode, struct file *flip);
static ssize_t device_read(struct file *flip, char *buffer, size_t length,loff_t *offset);
static ssize_t device_write(struct file *flip, char *buffer, size_t length, loff_t *offset);
static int is_buffer_empty(void);
static int is_buffer_full(void);
static int read_buffer_char(char *buffer);
static int write_buffer_char(char *buffer);

static struct file_operations device_fops =
{
	read:device_read,     /* read */
	write:device_write,    /* write */
	open:device_open,     /* open */
	release:device_release   /* release */
};

int init_module()
{
	printk("RAM Buffer Module loading...\n");
	if( ( s_nMajor = register_chrdev(0, DEVICE_NAME, &device_fops) )<0 )
	{
		printk(DEVICE_NAME" : Device Registration Failed ~! (%d) \n",s_nMajor);
		return s_nMajor;
	}
	printk(DEVICE_NAME" : Device Registration with Major Number = %d\n",s_nMajor);
	strcpy(s_strBuf,"안녕하세요 세상들이셔~!\n");
	s_nBufEnd = strlen(s_strBuf) + 1;
	return 0;
}

void cleanup_module(void)
{
	int nRetCode;
	printk("Unloading RAM Buffer Module\n");
	if( (nRetCode = unregister_chrdev(s_nMajor, DEVICE_NAME))<0 )
	{
		printk(DEVICE_NAME" : Device unregisteration failed (%d)\n",nRetCode);
	}
}

int device_open(struct inode *inode, struct file *flip)
{
	printk(DEVICE_NAME" : Device open (%d, %d)\n",MAJOR(inode->i_rdev), MINOR(inode->i_rdev) );
	if(s_bDeviceOpen)
	{
		printk(DEVICE_NAME" : Device already open\n");
		return -EBUSY;
	}
	++s_bDeviceOpen;
	MOD_INC_USE_COUNT;
	return 0;
}

int device_release(struct inode *inode, struct file *flip)
{
	printk(DEVICE_NAME" : Device Release (%d, %d)\n",MAJOR(inode->i_rdev), MINOR(inode->i_rdev) );
	if(!s_bDeviceOpen)
	{
		printk(DEVICE_NAME" : Device has not opened\n");
		return -EINVAL;
	}
	--s_bDeviceOpen;
	MOD_DEC_USE_COUNT;
	return 0;									
}

ssize_t device_read(struct file *filp, char *buffer, size_t length, loff_t *offset)
{
	int count = 0;
	if (is_buffer_empty()) {
		printk(DEVICE_NAME " : Read return EOF\n");
		return 0;
	} // end if
	while (!is_buffer_empty() && length > 1) {
		read_buffer_char(buffer);
		++buffer;
		--length;
		++count;
	} // end while
	put_user(0, buffer);
	++count;
	printk(DEVICE_NAME " : read %d bytes\n", count);
	return count;
} // end func

ssize_t device_write(struct file *filp, char *buffer, size_t length, loff_t *offset)
{
	return -ENOSYS;
} // end func

// Buffer Management
int is_buffer_empty() {
	return (s_nBufPos == s_nBufEnd) ? 1 : 0;
} // end func

int is_buffer_full() {
	int pos = s_nBufEnd + 1;
	if (pos == BUFFER_LEN) pos = 0;
	return (pos == s_nBufPos) ? 1 : 0;
} // end func

int read_buffer_char(char *buffer) {
	if (is_buffer_empty()) return -1;
	put_user(s_strBuf[s_nBufPos], buffer);
	if (++s_nBufPos == BUFFER_LEN) s_nBufPos = 0;
	return 0;
} // end func

int write_buffer_char(char *buffer) {
	if (is_buffer_full()) return -1;
	get_user(s_strBuf[s_nBufEnd], buffer);
	if (++s_nBufEnd == BUFFER_LEN) s_nBufEnd = 0;
	return 0;
} // end func

다름이 아니라 여기에 나온 put_user와 get_user부분에 관련된
manual을 알고 싶은데 man put_user로 해서도 찾을 수 없었습니다.
관련된 부분을 어떻게 찾을 수 있을지요?

그리고 아마 여기서 가장 중요한
함수인듯한데, 간단히라도 좋으니 put_user와
get_user의 기능을 알려주신다면 감사하겠습니다.

put_user : 커널 메모리의 내용을 User Process 메모리로 복사
get_user : User Process 메모리의 내용을 커널 메모리로 복사

이것입니다. 고운 하루되시길..

=========================
CharSyam ^^ --- 고운 하루
=========================

댓글 달기

Filtered HTML

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

BBCode

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param>
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

Textile

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • You can use Textile markup to format text.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Markdown

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • Quick Tips:
    • Two or more spaces at a line's end = Line break
    • Double returns = Paragraph
    • *Single asterisks* or _single underscores_ = Emphasis
    • **Double** or __double__ = Strong
    • This is [a link](http://the.link.example.com "The optional title text")
    For complete details on the Markdown syntax, see the Markdown documentation and Markdown Extra documentation for tables, footnotes, and more.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Plain text

  • HTML 태그를 사용할 수 없습니다.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 줄과 단락은 자동으로 분리됩니다.
댓글 첨부 파일
이 댓글에 이미지나 파일을 업로드 합니다.
파일 크기는 8 MB보다 작아야 합니다.
허용할 파일 형식: txt pdf doc xls gif jpg jpeg mp3 png rar zip.
CAPTCHA
이것은 자동으로 스팸을 올리는 것을 막기 위해서 제공됩니다.