diff options
Diffstat (limited to 'litmus/ftdev.c')
-rw-r--r-- | litmus/ftdev.c | 131 |
1 files changed, 91 insertions, 40 deletions
diff --git a/litmus/ftdev.c b/litmus/ftdev.c index d68d05c6d7dc..4a4b2e3e56c2 100644 --- a/litmus/ftdev.c +++ b/litmus/ftdev.c | |||
@@ -310,80 +310,131 @@ struct file_operations ftdev_fops = { | |||
310 | .read = ftdev_read, | 310 | .read = ftdev_read, |
311 | }; | 311 | }; |
312 | 312 | ||
313 | 313 | int ftdev_init( struct ftdev* ftdev, struct module* owner, | |
314 | void ftdev_init(struct ftdev* ftdev, struct module* owner, const char* name) | 314 | const int minor_cnt, const char* name) |
315 | { | 315 | { |
316 | int i, error; | 316 | int i, err; |
317 | |||
318 | BUG_ON(minor_cnt < 1); | ||
319 | |||
317 | cdev_init(&ftdev->cdev, &ftdev_fops); | 320 | cdev_init(&ftdev->cdev, &ftdev_fops); |
318 | ftdev->name = name; | 321 | ftdev->name = name; |
322 | ftdev->minor_cnt = minor_cnt; | ||
319 | ftdev->cdev.owner = owner; | 323 | ftdev->cdev.owner = owner; |
320 | ftdev->cdev.ops = &ftdev_fops; | 324 | ftdev->cdev.ops = &ftdev_fops; |
321 | ftdev->minor_cnt = 0; | 325 | ftdev->alloc = NULL; |
322 | for (i = 0; i < MAX_FTDEV_MINORS; i++) { | 326 | ftdev->free = NULL; |
327 | ftdev->can_open = NULL; | ||
328 | |||
329 | ftdev->minor = kcalloc(ftdev->minor_cnt, sizeof(*ftdev->minor), | ||
330 | GFP_KERNEL); | ||
331 | if (!ftdev->minor) { | ||
332 | printk(KERN_WARNING "ftdev(%s): Could not allocate memory\n", | ||
333 | ftdev->name); | ||
334 | err = -ENOMEM; | ||
335 | goto err_out; | ||
336 | } | ||
337 | |||
338 | for (i = 0; i < ftdev->minor_cnt; i++) { | ||
323 | mutex_init(&ftdev->minor[i].lock); | 339 | mutex_init(&ftdev->minor[i].lock); |
324 | ftdev->minor[i].readers = 0; | 340 | ftdev->minor[i].readers = 0; |
325 | ftdev->minor[i].buf = NULL; | 341 | ftdev->minor[i].buf = NULL; |
326 | ftdev->minor[i].events = NULL; | 342 | ftdev->minor[i].events = NULL; |
327 | } | 343 | } |
328 | ftdev->alloc = NULL; | ||
329 | ftdev->free = NULL; | ||
330 | ftdev->can_open = NULL; | ||
331 | 344 | ||
332 | ftdev->class = class_create(owner, ftdev->name); | 345 | ftdev->class = class_create(owner, ftdev->name); |
333 | if (IS_ERR(ftdev->class)) { | 346 | if (IS_ERR(ftdev->class)) { |
334 | error = PTR_ERR(ftdev->class); | 347 | err = PTR_ERR(ftdev->class); |
335 | printk(KERN_WARNING "ftdev(%s): " | 348 | printk(KERN_WARNING "ftdev(%s): " |
336 | "Could not create device class.\n", | 349 | "Could not create device class.\n", ftdev->name); |
337 | name); | 350 | goto err_dealloc; |
338 | } | 351 | } |
352 | |||
353 | return 0; | ||
354 | |||
355 | err_dealloc: | ||
356 | kfree(ftdev->minor); | ||
357 | err_out: | ||
358 | return err; | ||
359 | } | ||
360 | |||
361 | /* | ||
362 | * Destroy minor devices up to, but not including, up_to. | ||
363 | */ | ||
364 | static void ftdev_device_destroy(struct ftdev* ftdev, unsigned int up_to) | ||
365 | { | ||
366 | dev_t minor_cntr; | ||
367 | |||
368 | if (up_to < 1) | ||
369 | up_to = (ftdev->minor_cnt < 1) ? 0 : ftdev->minor_cnt; | ||
370 | |||
371 | for (minor_cntr = 0; minor_cntr < up_to; ++minor_cntr) | ||
372 | device_destroy(ftdev->class, MKDEV(ftdev->major, minor_cntr)); | ||
373 | } | ||
374 | |||
375 | void ftdev_exit(struct ftdev* ftdev) | ||
376 | { | ||
377 | printk("ftdev(%s): Exiting\n", ftdev->name); | ||
378 | ftdev_device_destroy(ftdev, -1); | ||
379 | cdev_del(&ftdev->cdev); | ||
380 | unregister_chrdev_region(MKDEV(ftdev->major, 0), ftdev->minor_cnt); | ||
381 | class_destroy(ftdev->class); | ||
382 | kfree(ftdev->minor); | ||
339 | } | 383 | } |
340 | 384 | ||
341 | int register_ftdev(struct ftdev* ftdev) | 385 | int register_ftdev(struct ftdev* ftdev) |
342 | { | 386 | { |
343 | struct device **device; | 387 | struct device **device; |
344 | dev_t trace_dev; | 388 | dev_t trace_dev_tmp, minor_cntr; |
345 | int error = 0, major, i; | 389 | int err; |
346 | 390 | ||
347 | error = alloc_chrdev_region(&trace_dev, 0, ftdev->minor_cnt, | 391 | err = alloc_chrdev_region(&trace_dev_tmp, 0, ftdev->minor_cnt, |
348 | ftdev->name); | 392 | ftdev->name); |
349 | major = MAJOR(trace_dev); | 393 | if (err) { |
350 | if (error) | ||
351 | { | ||
352 | printk(KERN_WARNING "ftdev(%s): " | 394 | printk(KERN_WARNING "ftdev(%s): " |
353 | "Could not register major/minor number %d/%u\n", | 395 | "Could not allocate char. device region (%d minors)\n", |
354 | ftdev->name, major, ftdev->minor_cnt); | 396 | ftdev->name, ftdev->minor_cnt); |
355 | goto out; | 397 | goto err_out; |
356 | } | 398 | } |
357 | error = cdev_add(&ftdev->cdev, trace_dev, ftdev->minor_cnt); | 399 | |
358 | if (error) { | 400 | ftdev->major = MAJOR(trace_dev_tmp); |
401 | |||
402 | err = cdev_add(&ftdev->cdev, trace_dev_tmp, ftdev->minor_cnt); | ||
403 | if (err) { | ||
359 | printk(KERN_WARNING "ftdev(%s): " | 404 | printk(KERN_WARNING "ftdev(%s): " |
360 | "Could not add cdev for major/minor = %d/%u.\n", | 405 | "Could not add cdev for major %u with %u minor(s).\n", |
361 | ftdev->name, major, ftdev->minor_cnt); | 406 | ftdev->name, ftdev->major, ftdev->minor_cnt); |
362 | goto out; | 407 | goto err_unregister; |
363 | } | 408 | } |
364 | 409 | ||
365 | /* | 410 | /* create the minor device(s) */ |
366 | * create all the minor devices | 411 | for (minor_cntr = 0; minor_cntr < ftdev->minor_cnt; ++minor_cntr) |
367 | */ | ||
368 | for (i = 0; i < ftdev->minor_cnt; ++i) | ||
369 | { | 412 | { |
370 | trace_dev = MKDEV(major, i); | 413 | trace_dev_tmp = MKDEV(ftdev->major, minor_cntr); |
371 | device = &(ftdev->minor[i].device); | 414 | device = &ftdev->minor[minor_cntr].device; |
372 | 415 | ||
373 | *device = device_create(ftdev->class, NULL, trace_dev, NULL, | 416 | *device = device_create(ftdev->class, NULL, trace_dev_tmp, NULL, |
374 | "%s%d", ftdev->name, i); | 417 | "litmus/%s%d", ftdev->name, minor_cntr); |
375 | if (IS_ERR(*device)) { | 418 | if (IS_ERR(*device)) { |
376 | error = PTR_ERR(*device); | 419 | err = PTR_ERR(*device); |
377 | printk(KERN_WARNING "ftdev(%s): " | 420 | printk(KERN_WARNING "ftdev(%s): " |
378 | "Could not create device major/minor number " | 421 | "Could not create device major/minor number " |
379 | "%d/%d\n", ftdev->name, major, i); | 422 | "%u/%u\n", ftdev->name, ftdev->major, |
423 | minor_cntr); | ||
380 | printk(KERN_WARNING "ftdev(%s): " | 424 | printk(KERN_WARNING "ftdev(%s): " |
381 | "Will not continue creating devices. Tracing " | 425 | "will attempt deletion of allocated devices.\n", |
382 | "may be in an inconsistent state.\n", | ||
383 | ftdev->name); | 426 | ftdev->name); |
384 | goto out; | 427 | goto err_minors; |
385 | } | 428 | } |
386 | } | 429 | } |
387 | out: | 430 | |
388 | return error; | 431 | return 0; |
432 | |||
433 | err_minors: | ||
434 | ftdev_device_destroy(ftdev, minor_cntr); | ||
435 | cdev_del(&ftdev->cdev); | ||
436 | err_unregister: | ||
437 | unregister_chrdev_region(MKDEV(ftdev->major, 0), ftdev->minor_cnt); | ||
438 | err_out: | ||
439 | return err; | ||
389 | } | 440 | } |