字符设备初始化流程



字符设备初始化流程:

1.

register_chrdev_region() - register a range of device numbers

alloc_chrdev_region() - register a range of char device numbers

1.1

Register a single major with a specified minor range.

2.

cdev_init() - initialize a cdev structure

3.

cdev_add() - add a char device to the system

 

static struct char_device_struct {

struct char_device_struct *next;

 

unsigned int major;

unsigned int baseminor;

int minorct;

char name[64];

struct cdev *cdev; /* will die */

} *chrdevs[CHRDEV_MAJOR_HASH_SIZE];

 

static int __init s3c2440_leds_init(void)

{

int result;

dev_t ledno = MKDEV(leds_major, 0);

char dev_name[] = "leds";

 

/*申请静态设备号*/

if (leds_major)

{

result = register_chrdev_region(ledno, 1, dev_name);

}

else

{/*动态分配设备号*/

result = alloc_chrdev_region(&ledno, 0, 1, dev_name);

leds_major = MAJOR(ledno);

}

if (result < 0)

{

printk(KERN_WARNING "leds:unable to get major %d\n"leds_major);

return result;

}

/*初始化cdev设备*/

cdev_init(&leds_dev, &leds_fops);

leds_dev.owner = THIS_MODULE;

leds_dev.ops = &leds_fops;

/*注册cdev设备*/

result = cdev_add(&leds_devMKDEV(leds_major, 0), 1);

if (result)

printk(KERN_NOTICE "Error %d adding leds\n"result);

printk("Leds device installed,with major %d\n"leds_major);

printk("The device name is :%s\n"dev_name);

/*自动创建设备文件*/

leds_class = class_create(THIS_MODULE"leds_driver");

device_create(leds_classNULLMKDEV(leds_major, 0), NULL"leds");

return 0;

}

 

 

/**

 * register_chrdev_region() - register a range of device numbers

 * @from: the first in the desired range of device numbers; must include

 *        the major number.

 * @count: the number of consecutive device numbers required

 * @name: the name of the device or driver.

 *

 * Return value is zero on success, a negative error code on failure.

 */

int register_chrdev_region(dev_t fromunsigned countconst char *name)

{

struct char_device_struct *cd;

dev_t to = from + count;

dev_t nnext;

 

for (n = fromn < ton = next) {

next = MKDEV(MAJOR(n)+1, 0);

if (next > to)

next = to;

cd = __register_chrdev_region(MAJOR(n), MINOR(n),

       next - nname);

if (IS_ERR(cd))

goto fail;

}

return 0;

fail:

to = n;

for (n = fromn < ton = next) {

next = MKDEV(MAJOR(n)+1, 0);

kfree(__unregister_chrdev_region(MAJOR(n), MINOR(n), next - n));

}

return PTR_ERR(cd);

}

 

/**

 * alloc_chrdev_region() - register a range of char device numbers

 * @dev: output parameter for first assigned number

 * @baseminor: first of the requested range of minor numbers

 * @count: the number of minor numbers required

 * @name: the name of the associated device or driver

 *

 * Allocates a range of char device numbers.  The major number will be

 * chosen dynamically, and returned (along with the first minor number)

 * in @dev.  Returns zero or a negative error code.

 */

int alloc_chrdev_region(dev_t *devunsigned baseminorunsigned count,

const char *name)

{

struct char_device_struct *cd;

cd = __register_chrdev_region(0, baseminorcountname);

if (IS_ERR(cd))

return PTR_ERR(cd);

*dev = MKDEV(cd->majorcd->baseminor);

return 0;

}

 

/*

 * Register a single major with a specified minor range.

 *

 * If major == 0 this functions will dynamically allocate a major and return

 * its number.

 *

 * If major > 0 this function will attempt to reserve the passed range of

 * minors and will return zero on success.

 *

 * Returns a -ve errno on failure.

 */

static struct char_device_struct *

__register_chrdev_region(unsigned int majorunsigned int baseminor,

   int minorctconst char *name)

{

struct char_device_struct *cd, **cp;

int ret = 0;

int i;

 

cd = kzalloc(sizeof(struct char_device_struct), GFP_KERNEL);

if (cd == NULL)

return ERR_PTR(-ENOMEM);

 

mutex_lock(&chrdevs_lock);

 

/* temporary */

if (major == 0) {

for (i = ARRAY_SIZE(chrdevs)-1; i > 0; i--) {

if (chrdevs[i] == NULL)

break;

}

 

if (i == 0) {

ret = -EBUSY;

goto out;

}

major = i;

ret = major;

}

 

cd->major = major;

cd->baseminor = baseminor;

cd->minorct = minorct;

strlcpy(cd->namenamesizeof(cd->name));

 

i = major_to_index(major);

 

for (cp = &chrdevs[i]; *cpcp = &(*cp)->next)

if ((*cp)->major > major ||

    ((*cp)->major == major &&

     (((*cp)->baseminor >= baseminor) ||

      ((*cp)->baseminor + (*cp)->minorct > baseminor))))

break;

 

/* Check for overlapping minor ranges.  */

if (*cp && (*cp)->major == major) {

int old_min = (*cp)->baseminor;

int old_max = (*cp)->baseminor + (*cp)->minorct - 1;

int new_min = baseminor;

int new_max = baseminor + minorct - 1;

 

/* New driver overlaps from the left.  */

if (new_max >= old_min && new_max <= old_max) {

ret = -EBUSY;

goto out;

}

 

/* New driver overlaps from the right.  */

if (new_min <= old_max && new_min >= old_min) {

ret = -EBUSY;

goto out;

}

}

 

cd->next = *cp;

*cp = cd;

mutex_unlock(&chrdevs_lock);

return cd;

out:

mutex_unlock(&chrdevs_lock);

kfree(cd);

return ERR_PTR(ret);

}

 

/**

 * cdev_init() - initialize a cdev structure

 * @cdev: the structure to initialize

 * @fops: the file_operations for this device

 *

 * Initializes @cdev, remembering @fops, making it ready to add to the

 * system with cdev_add().

 */

void cdev_init(struct cdev *cdevconst struct file_operations *fops)

{

memset(cdev, 0, sizeof *cdev);

INIT_LIST_HEAD(&cdev->list);

kobject_init(&cdev->kobj, &ktype_cdev_default);

cdev->ops = fops;

}

/**

 * kobject_init - initialize a kobject structure

 * @kobj: pointer to the kobject to initialize

 * @ktype: pointer to the ktype for this kobject.

 *

 * This function will properly initialize a kobject such that it can then

 * be passed to the kobject_add() call.

 *

 * After this function is called, the kobject MUST be cleaned up by a call

 * to kobject_put(), not by a call to kfree directly to ensure that all of

 * the memory is cleaned up properly.

 */

void kobject_init(struct kobject *kobjstruct kobj_type *ktype)

{

char *err_str;

 

if (!kobj) {

err_str = "invalid kobject pointer!";

goto error;

}

if (!ktype) {

err_str = "must have a ktype to be initialized properly!\n";

goto error;

}

if (kobj->state_initialized) {

/* do not error out as sometimes we can recover */

printk(KERN_ERR "kobject (%p): tried to init an initialized "

       "object, something is seriously wrong.\n"kobj);

dump_stack();

}

 

kobject_init_internal(kobj);

kobj->ktype = ktype;

return;

 

error:

printk(KERN_ERR "kobject (%p): %s\n"kobjerr_str);

dump_stack();

}

 

/**

 * cdev_add() - add a char device to the system

 * @p: the cdev structure for the device

 * @dev: the first device number for which this device is responsible

 * @count: the number of consecutive minor numbers corresponding to this

 *         device

 *

 * cdev_add() adds the device represented by @p to the system, making it

 * live immediately.  A negative error code is returned on failure.

 */

int cdev_add(struct cdev *pdev_t devunsigned count)

{

p->dev = dev;

p->count = count;

return kobj_map(cdev_mapdevcountNULLexact_matchexact_lockp);

}

 

int kobj_map(struct kobj_map *domaindev_t devunsigned long range,

     struct module *modulekobj_probe_t *probe,

     int (*lock)(dev_tvoid *), void *data)

{

unsigned n = MAJOR(dev + range - 1) - MAJOR(dev) + 1;

unsigned index = MAJOR(dev);

unsigned i;

struct probe *p;

 

if (n > 255)

n = 255;

 

p = kmalloc(sizeof(struct probe) * nGFP_KERNEL);

 

if (p == NULL)

return -ENOMEM;

 

for (i = 0; i < ni++, p++) {

p->owner = module;

p->get = probe;

p->lock = lock;

p->dev = dev;

p->range = range;

p->data = data;

}

mutex_lock(domain->lock);

for (i = 0, p -= ni < ni++, p++, index++) {

struct probe **s = &domain->probes[index % 255];

while (*s && (*s)->range < range)

s = &(*s)->next;

p->next = *s;

*s = p;

}

mutex_unlock(domain->lock);

return 0;

}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值