Newer
Older
/* Federico Vaga for CERN, 2011, GNU GPLv2 or later */
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/types.h>
#include <linux/slab.h>
#include <linux/fs.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/spinlock.h>
#define __ZIO_INTERNAL__
#include <linux/zio.h>
#include <linux/zio-buffer.h>
#include <linux/zio-trigger.h>
static struct zio_status *zstat = &zio_global_status; /* Always use ptr */
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
const char zio_attr_names[ZATTR_STD_ATTR_NUM][ZIO_NAME_LEN] = {
[ZATTR_GAIN] = "gain_factor",
[ZATTR_OFFSET] = "offset",
[ZATTR_NBIT] = "resolution_bits",
[ZATTR_MAXRATE] = "max_sample_rate",
[ZATTR_VREFTYPE] = "vref_src",
};
EXPORT_SYMBOL(zio_attr_names);
/*
* @zattrs_to_attrs: extract a 'struct attribute **' from a
* 'struct zio_attribute *' and return it
*/
static struct attribute **zattrs_to_attrs(struct zio_attribute *zattr,
unsigned int n_attr)
{
int i;
struct attribute **attrs;
pr_debug("%s\n", __func__);
if (!zattr || !n_attr)
return NULL;
attrs = kzalloc(sizeof(struct attribute) * n_attr, GFP_KERNEL);
if (!attrs)
return NULL;
for (i = 0; i < n_attr; i++)
attrs[i] = &zattr[i].attr;
return attrs;
}
/*
* Top-level ZIO objects has a unique name.
* You can find a particular object by searching its name.
*/
static inline struct zio_object_list_item *__find_by_name(
struct zio_object_list *zobj_list, char *name)
{
struct zio_object_list_item *cur;
list_for_each_entry(cur, &zobj_list->list, list) {
if (strcmp(cur->obj_head->name, name) == 0)
return cur; /* object found */
/* Find a buffer type from its name */
static struct zio_buffer_type *zbuf_find_by_name(char *name)
{
struct zio_object_list_item *list_item;
list_item = __find_by_name(&zstat->all_buffer_types, name);
if (!list_item)
return container_of(list_item->obj_head, struct zio_buffer_type, head);
/* Find a trigger type from its name */
static struct zio_trigger_type *trig_find_by_name(char *name)
{
struct zio_object_list_item *list_item;
list_item = __find_by_name(&zstat->all_trigger_types, name);
if (!list_item)
return container_of(list_item->obj_head, struct zio_trigger_type, head);
static int __zio_fire_input_trigger(struct zio_ti *ti)
{
struct zio_buffer_type *zbuf;
struct zio_block *block;
struct zio_device *zdev;
struct zio_cset *cset;
struct zio_channel *chan;
struct zio_control *ctrl;
int err;
cset = ti->cset;
zdev = cset->zdev;
zbuf = cset->zbuf;
pr_debug("%s:%d\n", __func__, __LINE__);
cset_for_each(cset, chan) {
/* Allocate the buffer for the incoming sample */
ctrl = zio_alloc_control(GFP_ATOMIC);
return -ENOMEM;
}
memcpy(ctrl, ti->current_ctrl, ZIO_CONTROL_SIZE);
ctrl->chan_i = chan->index;
block = zbuf->b_op->alloc_block(chan->bi, ctrl,
ctrl->ssize * ctrl->nsamples,
GFP_ATOMIC);
if (IS_ERR(block)) {
zio_free_control(ctrl);
return PTR_ERR(block);
/* Get samples, and control block, then store it*/
err = zdev->d_op->input_block(chan, block);
if (err) {
pr_err("%s: input_block(%s:%i:%i) error %d\n", __func__,
chan->cset->zdev->head.name,
chan->cset->index,
chan->index,
err);
zbuf->b_op->free_block(chan->bi, block);
err = zbuf->b_op->store_block(chan->bi, block);
if (err) {
/* No error message for common error */
zbuf->b_op->free_block(chan->bi, block);
}
static int __zio_fire_output_trigger(struct zio_ti *ti)
{
struct zio_buffer_type *zbuf;
struct zio_block *block;
struct zio_device *zdev;
struct zio_cset *cset;
struct zio_channel *chan;
int err = 0;
cset = ti->cset;
zdev = cset->zdev;
zbuf = cset->zbuf;
pr_debug("%s:%d\n", __func__, __LINE__);
cset_for_each(cset, chan) {
/* Users of zio_fire_trigger must store a block in t_priv */
block = chan->t_priv;
if (!block) /* And some channel may be missing data */
continue;
err = zdev->d_op->output_block(chan, block);
if (err) {
pr_err("%s: output_block(%s:%i:%i) error %d\n",
__func__,
chan->cset->zdev->head.name,
chan->cset->index,
chan->index,
err);
}
/* Error or not, free the block and proceed */
zbuf->b_op->free_block(chan->bi, block);
/* We may have a new block ready or not */
chan->t_priv = zbuf->b_op->retr_block(chan->bi);
}
return 0;
}
/*
* When a software trigger fires, it should call this function. Hw ones don't
*/
int zio_fire_trigger(struct zio_ti *ti)
{
/* If the trigger runs too early, ti->cset is still NULL */
if (!ti->cset)
return -EAGAIN;
/* Copy the stamp (we are software driven anyways) */
ti->current_ctrl->tstamp.secs = ti->tstamp.tv_sec;
ti->current_ctrl->tstamp.ticks = ti->tstamp.tv_nsec;
ti->current_ctrl->tstamp.bins = ti->tstamp_extra;
/*
* And the sequence number too (first returned seq is 1).
* Sequence number is always increased to identify un-stored
* blocks or other errors in trigger activation.
*/
ti->current_ctrl->seq_num++;
if (likely((ti->flags & ZIO_DIR) == ZIO_DIR_INPUT))
return __zio_fire_input_trigger(ti);
return __zio_fire_output_trigger(ti);
}
EXPORT_SYMBOL(zio_fire_trigger);
static int __has_auto_index(char *s)
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
{
int i = 0;
for (i = 0; i < ZIO_NAME_LEN-1; i++) {
if (s[i] != '%')
continue;
i++;
if (s[i] == 'd')
return 1;
}
return 0;
}
static int __next_strlen(char *str)
{
int increment = 0, i;
for (i = strlen(str)-1; i > 0; i--) {
/* if is an ascii number */
if (str[i] >= '0' && str[i] <= '9') {
if (str[i] == '9')
continue;
else
break;
} else {
increment++;
break;
}
}
return strlen(str) + increment;
}
/*
* The zio device name must be unique. If it is not unique, a busy error is
* returned.
*/
static int zobj_unique_name(struct zio_object_list *zobj_list, char *name)
{
struct zio_object_list_item *cur;
struct zio_obj_head *tmp;
unsigned int counter = 0, again, len;
char name_to_check[ZIO_NAME_LEN];
int auto_index = __has_auto_index(name);
pr_debug("%s\n", __func__);
if (!name)
return -EINVAL;
len = strlen(name);
if (!len)
return -EINVAL;
strncpy(name_to_check, name, ZIO_NAME_LEN);
do {
again = 0;
if (auto_index) { /* TODO when zio become bus, it is useless */
sprintf(name_to_check, name, counter++);
len = strlen(name_to_check);
}
list_for_each_entry(cur, &zobj_list->list, list) {
tmp = cur->obj_head;
if (strcmp(tmp->name, name_to_check))
continue; /* no conflict */
/* conflict found */
/* if not auto-assigned, then error */
if (!auto_index) {
pr_err("ZIO: name \"%s\" is already taken\n",
name);
return -EBUSY;
}
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
if (__next_strlen(name_to_check) > ZIO_NAME_LEN) {
pr_err("ZIO: invalid name \"%s\"\n", name);
return -EINVAL;
}
again = 1;
break;
}
} while (again);
strncpy(name, name_to_check, ZIO_NAME_LEN);
return 0;
}
static struct zio_attribute *__zattr_clone(const struct zio_attribute *src,
unsigned int n)
{
struct zio_attribute *dest = NULL;
unsigned int size;
if (!src)
return NULL;
size = n * sizeof(struct zio_attribute);
dest = kmalloc(size, GFP_KERNEL);
if (!dest)
return ERR_PTR(-ENOMEM);
dest = memcpy(dest, src, size);
return dest;
}
static int zattr_chan_pre_set(struct zio_channel *chan)
{
struct zio_cset *cset = chan->cset;
if (!(cset->flags & ZCSET_CHAN_ALLOC))
return 0; /* nothing to do */
/*
* If the channel has been allocated by ZIO, then attributes are
* cloned from the template channel description within parent cset
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
*/
chan->zattr_set.std_zattr =
__zattr_clone(
cset->zattr_set_chan.std_zattr,
ZATTR_STD_ATTR_NUM);
if (IS_ERR(chan->zattr_set.std_zattr))
return PTR_ERR(chan->zattr_set.std_zattr);
chan->zattr_set.ext_zattr =
__zattr_clone(
cset->zattr_set.ext_zattr,
cset->zattr_set.n_ext_attr);
if (IS_ERR(chan->zattr_set.ext_zattr)) {
kfree(chan->zattr_set.std_zattr);
return PTR_ERR(chan->zattr_set.ext_zattr);
}
return 0;
}
static void zattr_chan_post_remove(struct zio_channel *chan)
{
if (chan->cset->flags & ZCSET_CHAN_ALLOC) {
kfree(chan->zattr_set.std_zattr);
kfree(chan->zattr_set.ext_zattr);
}
}
/* When touching attributes, we always use the spinlock for the hosting dev */
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
static spinlock_t *zdev_get_spinlock(struct zio_obj_head *head)
{
spinlock_t *lock;
switch (head->zobj_type) {
case ZDEV:
lock = &to_zio_dev(&head->kobj)->lock;
break;
case ZCSET:
lock = &to_zio_cset(&head->kobj)->zdev->lock;
break;
case ZCHAN:
lock = &to_zio_chan(&head->kobj)->cset->zdev->lock;
break;
case ZTI: /* we might not want to take a lock but... */
lock = &to_zio_ti(&head->kobj)->cset->zdev->lock;
break;
case ZBI:
lock = &to_zio_bi(&head->kobj)->cset->zdev->lock;
break;
default:
WARN(1, "ZIO: unknown zio object %i\n", head->zobj_type);
return NULL;
}
return lock;
}
/* Retrieve an attribute set from an object head */
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
static struct zio_attribute_set *__get_zattr_set(struct zio_obj_head *head)
{
struct zio_attribute_set *zattr_set;
switch (head->zobj_type) {
case ZDEV:
zattr_set = &to_zio_dev(&head->kobj)->zattr_set;
break;
case ZCSET:
zattr_set = &to_zio_cset(&head->kobj)->zattr_set;
break;
case ZCHAN:
zattr_set = &to_zio_chan(&head->kobj)->zattr_set;
break;
case ZTI:
zattr_set = &to_zio_ti(&head->kobj)->zattr_set;
break;
case ZBI:
zattr_set = &to_zio_bi(&head->kobj)->zattr_set;
break;
default:
WARN(1, "ZIO: unknown zio object %i\n", head->zobj_type);
return NULL;
}
return zattr_set;
}
/*
* Zio objects all handle uint32_t values. So the show and store
* are centralized here, and each device has its own get_info and set_conf
* which handle binary 32-bit numbers. Both the function are locked to prevent
* concurrency issue when editing device register.
*/
static ssize_t zio_attr_show(struct kobject *kobj, struct attribute *attr,
char *buf)
{
int err = 0;
ssize_t len = 0;
spinlock_t *lock;
struct zio_attribute *zattr = to_zio_zattr(attr);
pr_debug("%s\n", __func__);
if (unlikely(strcmp(attr->name, "name") == 0)) {
/* print device name*/
return sprintf(buf, "%s\n", to_zio_head(kobj)->name);
}
if (zattr->s_op->info_get) {
lock = zdev_get_spinlock(to_zio_head(kobj));
spin_lock(lock);
err = zattr->s_op->info_get(kobj, zattr, &zattr->value);
spin_unlock(lock);
if (err)
return err;
}
len = sprintf(buf, "%i\n", zattr->value);
return len;
}
static ssize_t zio_attr_store(struct kobject *kobj, struct attribute *attr,
const char *buf, size_t size)
{
long val;
int err = 0;
struct zio_attribute *zattr = to_zio_zattr(attr);
spinlock_t *lock;
pr_debug("%s\n", __func__);
err = strict_strtol(buf, 10, &val);
if (err)
return -EINVAL;
if (zattr->s_op->conf_set) {
lock = zdev_get_spinlock(to_zio_head(kobj));
spin_lock(lock);
err = zattr->s_op->conf_set(kobj, zattr, val);
spin_unlock(lock);
}
return err == 0 ? size : err;
}
static const struct sysfs_ops zio_attribute_ktype_ops = {
.show = zio_attr_show,
.store = zio_attr_store,
};
static struct attribute default_attrs[] = {
.name = "name", /* show the name */
.mode = 0444, /* read only */
},
};
static struct attribute *def_attr_ptr[] = {
static struct kobj_type zdktype = { /* For standard and extended attribute */
.release = NULL,
.sysfs_ops = &zio_attribute_ktype_ops,
.default_attrs = def_attr_ptr,
};
static mode_t zattr_is_visible(struct kobject *kobj, struct attribute *attr,
int n)
{
unsigned int flag1, flag2, flag3;
mode_t mode = attr->mode;
/*
* FIXME: if it's decided that activation
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
* is always the first bit then is faster doing:
* flag1 & flag2 & flag3 & 0x1
* to verify content
*/
switch (__zio_get_object_type(kobj)) {
case ZDEV:
flag1 = to_zio_dev(kobj)->flags;
if (flag1 & ZIO_DISABLED)
mode = 0;
break;
case ZCSET:
flag1 = to_zio_cset(kobj)->flags;
flag2 = to_zio_cset(kobj)->zdev->flags;
if ((flag1 | flag2) & ZIO_DISABLED)
mode = 0;
break;
case ZCHAN:
flag1 = to_zio_chan(kobj)->flags;
flag2 = to_zio_chan(kobj)->cset->flags;
flag3 = to_zio_chan(kobj)->cset->zdev->flags;
if ((flag1 | flag2 | flag3) & ZIO_DISABLED)
mode = 0;
break;
default:
WARN(1, "ZIO: unknown zio object %i\n",
__zio_get_object_type(kobj));
}
return mode;
}
/*
* Verify attributes within the group,
* If they are valid register the group
*/
static int zattr_create_group(struct kobject *kobj,
struct attribute_group *grp, unsigned int n_attr,
const struct zio_sys_operations *s_op, int is_ext)
{
int i;
if (!grp->attrs)
return 0;
grp->is_visible = zattr_is_visible;
for (i = 0; i < n_attr; i++) {
/* Assign show and store function */
to_zio_zattr(grp->attrs[i])->s_op = s_op;
pr_warning("%s: can't create ext attributes. "
"%ith attribute has not a name", __func__, i);
return 0;
}
* Only standard attributes need these lines to fill
* the empty hole in the array of attributes
*/
grp->attrs[i]->name = zio_attr_names[i];
grp->attrs[i]->mode = 0;
}
}
return sysfs_create_group(kobj, grp);
}
/* Create a set of zio attributes: the standard one and the extended one */
static int zattr_create_set(struct zio_obj_head *head,
const struct zio_sys_operations *s_op)
{
int err = 0;
struct zio_attribute_set *zattr_set;
zattr_set = __get_zattr_set(head);
if (!zattr_set)
return -EINVAL; /* message already printed */
/* Create the standard attributes from zio attributes */
zattr_set->std_attr.attrs = zattrs_to_attrs(zattr_set->std_zattr,
ZATTR_STD_ATTR_NUM);
err = zattr_create_group(&head->kobj, &zattr_set->std_attr,
ZATTR_STD_ATTR_NUM, s_op, 0);
/* Create the extended attributes from zio attributes */
zattr_set->ext_attr.attrs = zattrs_to_attrs(zattr_set->ext_zattr,
zattr_set->n_ext_attr);
err = zattr_create_group(&head->kobj, &zattr_set->ext_attr,
zattr_set->n_ext_attr, s_op, 1);
if (err && zattr_set->std_attr.attrs)
sysfs_remove_group(&head->kobj, &zattr_set->std_attr);
out:
return err;
}
/* Remove an existing set of attribute */
static void zattr_remove_set(struct zio_obj_head *head)
{
struct zio_attribute_set *zattr_set;
zattr_set = __get_zattr_set(head);
if (!zattr_set)
return;
/* remove standard and extended attributes */
if (zattr_set->std_attr.attrs)
sysfs_remove_group(&head->kobj, &zattr_set->std_attr);
if (zattr_set->ext_attr.attrs)
sysfs_remove_group(&head->kobj, &zattr_set->ext_attr);
}
/* Create a buffer instance according to the buffer type defined in cset */
static int __buffer_create_instance(struct zio_channel *chan)
{
struct zio_buffer_type *zbuf = chan->cset->zbuf;
struct zio_bi *bi;
int err;
/* create buffer */
bi = zbuf->b_op->create(zbuf, chan, FMODE_READ);
if (IS_ERR(bi))
return PTR_ERR(bi);
/* Now fill the trigger instance, ops, head, then the rest */
bi->b_op = zbuf->b_op;
bi->f_op = zbuf->f_op;
bi->flags |= (chan->flags & ZIO_DIR);
bi->head.zobj_type = ZBI;
err = kobject_init_and_add(&bi->head.kobj, &zdktype,
&chan->head.kobj, "buffer");
if (err)
goto out_kobj;
snprintf(bi->head.name, ZIO_NAME_LEN, "%s-%s-%d-%d",
zbuf->head.name,
chan->cset->zdev->head.name,
chan->cset->index,
chan->index);
err = zattr_create_set(&bi->head, chan->cset->zdev->s_op);
if (err)
goto out_sysfs;
init_waitqueue_head(&bi->q);
spin_lock(&zbuf->lock);
list_add(&bi->list, &zbuf->list);
spin_unlock(&zbuf->lock);
bi->cset = chan->cset;
chan->bi = bi;
/* Done. This cset->ti marks everything is running (FIXME?) */
mb();
bi->chan = chan;
return 0;
out_sysfs:
kobject_del(&bi->head.kobj);
out_kobj:
kobject_put(&bi->head.kobj);
zbuf->b_op->destroy(bi);
return err;
}
/* Destroy a buffer instance */
static void __buffer_destroy_instance(struct zio_channel *chan)
{
struct zio_buffer_type *zbuf = chan->cset->zbuf;
struct zio_bi *bi = chan->bi;
chan->bi = NULL;
/* Remove from buffer instance list */
spin_lock(&zbuf->lock);
list_del(&bi->list);
spin_unlock(&zbuf->lock);
zattr_remove_set(&bi->head);
kobject_del(&bi->head.kobj);
kobject_put(&bi->head.kobj);
/* Finally destroy the instance */
zbuf->b_op->destroy(bi);
}
/* Create a trigger instance according to the trigger type defined in cset */
static int __trigger_create_instance(struct zio_cset *cset)
{
int err;
struct zio_control *ctrl;
struct zio_ti *ti;
pr_debug("%s:%d\n", __func__, __LINE__);
/* Allocate and fill current control as much as possible*/
ctrl = zio_alloc_control(GFP_KERNEL);
if (!ctrl)
ctrl->cset_i = cset->index;
strncpy(ctrl->devname, cset->zdev->head.name, ZIO_NAME_LEN);
strncpy(ctrl->triggername, cset->trig->head.name, ZIO_NAME_LEN);
ctrl->sbits = cset->ssize * 8; /* FIXME: retrieve from attribute */
ctrl->ssize = cset->ssize;
ti = cset->trig->t_op->create(cset->trig, cset, ctrl, 0/*FIXME*/);
if (IS_ERR(ti)) {
err = PTR_ERR(ti);
pr_err("%s: can't create trigger error %i\n", __func__, err);
goto out;
}
/* Now fill the trigger instance, ops, head, then the rest */
ti->t_op = cset->trig->t_op;
ti->f_op = cset->trig->f_op;
ti->flags |= cset->flags & ZIO_DIR;
ti->head.zobj_type = ZTI;
err = kobject_init_and_add(&ti->head.kobj, &zdktype,
&cset->head.kobj, "trigger");
if (err)
goto out_kobj;
snprintf(ti->head.name, ZIO_NAME_LEN, "%s-%s-%d",
cset->trig->head.name,
cset->zdev->head.name,
cset->index);
err = zattr_create_set(&ti->head, cset->zdev->s_op);
if (err)
goto out_sysfs;
/* Add to trigger instance list */
spin_lock(&cset->trig->lock);
list_add(&ti->list, &cset->trig->list);
spin_unlock(&cset->trig->lock);
cset->ti = ti;
/* Done. This cset->ti marks everything is running (FIXME?) */
mb();
ti->cset = cset;
return 0;
out_sysfs:
kobject_del(&ti->head.kobj);
out_kobj:
kobject_put(&ti->head.kobj);
ti->t_op->destroy(ti);
out:
zio_free_control(ctrl);
return err;
}
/* Destroy a buffer instance */
static void __trigger_destroy_instance(struct zio_cset *cset)
{
struct zio_ti *ti = cset->ti;
struct zio_control *ctrl = ti->current_ctrl;
cset->ti = NULL;
/* Remove from trigger instance list */
spin_lock(&cset->trig->lock);
list_del(&ti->list);
spin_unlock(&cset->trig->lock);
zattr_remove_set(&ti->head);
kobject_del(&ti->head.kobj);
kobject_put(&ti->head.kobj);
/* Finally destroy the instance and free the default control*/
cset->trig->t_op->destroy(ti);
zio_free_control(ctrl);
}
/*
* chan_register registers one channel. It is important to register
* or unregister all the channels of a cset at the same time to prevent
*/
static int chan_register(struct zio_channel *chan)
{
int err;
pr_debug("%s:%d\n", __func__, __LINE__);
if (!chan)
return -EINVAL;
chan->head.zobj_type = ZCHAN;
err = kobject_init_and_add(&chan->head.kobj, &zdktype,
&chan->cset->head.kobj, "chan%i", chan->index);
if (err)
goto out_add;
err = zattr_chan_pre_set(chan);
if (err)
goto out_pre;
/* Create sysfs channel attributes */
err = zattr_create_set(&chan->head, chan->cset->zdev->s_op);
if (err)
goto out_sysfs;
err = __buffer_create_instance(chan);
if (err)
goto out_buf;
err = zio_create_chan_devices(chan);
if (err)
goto out_create;
/*
* If no name was assigned, ZIO assigns it. channel name is
* set to the kobject name. kobject name has no length limit,
* so the channel name is the first ZIO_NAME_LEN characters of
* kobject name. A duplicate channel name is not a problem
* anyways.
*/
if (!strlen(chan->head.name))
strncpy(chan->head.name, chan->head.kobj.name, ZIO_NAME_LEN);
return 0;
out_create:
__buffer_destroy_instance(chan);
out_buf:
zattr_remove_set(&chan->head);
out_sysfs:
zattr_chan_post_remove(chan);
out_pre:
kobject_del(&chan->head.kobj);
out_add:
/* we must _put even if it returned error */
kobject_put(&chan->head.kobj);
return err;
}
static void chan_unregister(struct zio_channel *chan)
{
pr_debug("%s:%d\n", __func__, __LINE__);
if (!chan)
return;
zio_destroy_chan_devices(chan);
__buffer_destroy_instance(chan);
/* remove sysfs cset attributes */
zattr_remove_set(&chan->head);
zattr_chan_post_remove(chan);
kobject_del(&chan->head.kobj);
kobject_put(&chan->head.kobj);
}
/*
* @cset_alloc_chan: low-level drivers can avoid allocating their channels,
* they say how many are there and ZIO allocates them.
* @cset_free_chan: if ZIO allocated channels, then it frees them; otherwise
* it does nothing.
*/
static struct zio_channel *cset_alloc_chan(struct zio_cset *cset)
{
pr_debug("%s:%d\n", __func__, __LINE__);
/*if no static channels, then ZIO must alloc them */
if (cset->chan)
return cset->chan;
/* initialize memory to zero to have correct flags and attrs */
cset->chan = kzalloc(sizeof(struct zio_channel) *
cset->n_chan, GFP_KERNEL);
if (!cset->chan)
return ERR_PTR(-ENOMEM);
cset->flags |= ZCSET_CHAN_ALLOC;
return cset->chan;
}
static inline void cset_free_chan(struct zio_cset *cset)
{
pr_debug("%s:%d\n", __func__, __LINE__);
/* Only allocated channels need to be freed */
if (cset->flags & ZCSET_CHAN_ALLOC) {
kfree(cset->chan);
}
}
static int cset_register(struct zio_cset *cset)
{
int i, j, err = 0;
pr_debug("%s:%d\n", __func__, __LINE__);
if (!cset)
return -EINVAL;
if (!cset->n_chan) {
pr_err("ZIO: no channels in cset%i\n", cset->index);
return -EINVAL;
}
if (!cset->ssize) {
pr_err("ZIO: ssize can not be 0 in cset%i\n", cset->index);
return -EINVAL;
}
err = __zio_minorbase_get(cset);
if (err) {
pr_err("ZIO: no minors available\n");
return -EBUSY;
}
cset->head.zobj_type = ZCSET;
err = kobject_init_and_add(&cset->head.kobj, &zdktype,
&cset->zdev->head.kobj, "cset%i", cset->index);
if (err)
goto out_add;
/* Create sysfs cset attributes */
err = zattr_create_set(&cset->head, cset->zdev->s_op);
if (err)
goto out_sysfs;
cset->chan = cset_alloc_chan(cset);
if (IS_ERR(cset->chan)) {
err = PTR_ERR(cset->chan);
goto out_alloc;
}
/*
* The cset must have a buffer type. If none is associated
* to the cset, ZIO selectes the default one.
*/
if (!cset->zbuf) {
cset->zbuf = zbuf_find_by_name(ZIO_DEFAULT_BUFFER);
if (!cset->zbuf) {
pr_err("ZIO: can't find buffer \"%s\"\n",
ZIO_DEFAULT_BUFFER);
err = -EBUSY;
goto out_buf;
}
}
for (i = 0; i < cset->n_chan; i++) {
cset->chan[i].index = i;
cset->chan[i].cset = cset;
cset->chan[i].flags |= cset->flags & ZIO_DIR;
err = chan_register(&cset->chan[i]);
if (err)
goto out_reg;
}
/*
* If no name was assigned, ZIO assigns it. cset name is
* set to the kobject name. kobject name has no length limit,
* so the cset name is the first ZIO_NAME_LEN characters of
* kobject name. A duplicate cset name is not a problem
* anyways.
*/
if (!strlen(cset->head.name))
strncpy(cset->head.name, cset->head.kobj.name, ZIO_NAME_LEN);
/*
* The cset must have a trigger type. If none is associated
* to the cset, ZIO selectes the default one.
* This is done late because each channel must be ready when
* the trigger fires.
*/
if (!cset->trig) {
cset->trig = trig_find_by_name(ZIO_DEFAULT_TRIGGER);
if (!cset->trig) {
pr_err("ZIO: can't find trigger \"%s\"\n",
ZIO_DEFAULT_TRIGGER);
err = -EBUSY;
goto out_reg;
}
err = __trigger_create_instance(cset);
if (err)
goto out_trig;
}
list_add(&cset->list_cset, &zstat->list_cset);
/* Private initialization function */
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
if (cset->init) {
err = cset->init(cset);
if (err)
goto out_init;
}
return 0;
out_init:
cset->trig->t_op->destroy(cset->ti);
out_trig:
cset->trig = NULL;
out_reg:
for (j = i-1; j >= 0; j--)
chan_unregister(&cset->chan[i]);
out_buf:
cset_free_chan(cset);
out_alloc:
zattr_remove_set(&cset->head);
out_sysfs:
kobject_del(&cset->head.kobj);
out_add:
/* we must _put even if it returned error */
kobject_put(&cset->head.kobj);
return err;
}
static void cset_unregister(struct zio_cset *cset)
{
int i;
pr_debug("%s:%d\n", __func__, __LINE__);
if (!cset)
return;
if (cset->exit)
cset->exit(cset);
__trigger_destroy_instance(cset);
cset->trig = NULL;
/* Unregister all child channels */
for (i = 0; i < cset->n_chan; i++)
chan_unregister(&cset->chan[i]);
cset->zbuf = NULL;
cset_free_chan(cset);
zattr_remove_set(&cset->head);
kobject_del(&cset->head.kobj);