file.c 10.7 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
T
Tejun Heo 已提交
2 3 4 5 6 7 8 9 10
 * fs/sysfs/file.c - sysfs regular (text) file implementation
 *
 * Copyright (c) 2001-3 Patrick Mochel
 * Copyright (c) 2007 SUSE Linux Products GmbH
 * Copyright (c) 2007 Tejun Heo <teheo@suse.de>
 *
 * This file is released under the GPLv2.
 *
 * Please see Documentation/filesystems/sysfs.txt for more information.
L
Linus Torvalds 已提交
11 12 13 14
 */

#include <linux/module.h>
#include <linux/kobject.h>
15
#include <linux/kallsyms.h>
16
#include <linux/slab.h>
17
#include <linux/list.h>
18
#include <linux/mutex.h>
19
#include <linux/seq_file.h>
L
Linus Torvalds 已提交
20 21

#include "sysfs.h"
22
#include "../kernfs/kernfs-internal.h"
T
Tejun Heo 已提交
23

T
Tejun Heo 已提交
24
/*
25
 * Determine ktype->sysfs_ops for the given kernfs_node.  This function
T
Tejun Heo 已提交
26 27
 * must be called while holding an active reference.
 */
28
static const struct sysfs_ops *sysfs_file_ops(struct kernfs_node *kn)
T
Tejun Heo 已提交
29
{
30
	struct kobject *kobj = kn->parent->priv;
T
Tejun Heo 已提交
31

T
Tejun Heo 已提交
32
	if (kn->flags & KERNFS_LOCKDEP)
33
		lockdep_assert_held(kn);
T
Tejun Heo 已提交
34 35 36
	return kobj->ktype ? kobj->ktype->sysfs_ops : NULL;
}

37 38 39 40
/*
 * Reads on sysfs are handled through seq_file, which takes care of hairy
 * details like buffering and seeking.  The following function pipes
 * sysfs_ops->show() result through seq_file.
L
Linus Torvalds 已提交
41
 */
42
static int sysfs_kf_seq_show(struct seq_file *sf, void *v)
L
Linus Torvalds 已提交
43
{
44
	struct kernfs_open_file *of = sf->private;
45
	struct kobject *kobj = of->kn->parent->priv;
46
	const struct sysfs_ops *ops = sysfs_file_ops(of->kn);
L
Linus Torvalds 已提交
47
	ssize_t count;
48
	char *buf;
L
Linus Torvalds 已提交
49

50
	/* acquire buffer and ensure that it's >= PAGE_SIZE and clear */
51 52 53 54 55
	count = seq_get_buf(sf, &buf);
	if (count < PAGE_SIZE) {
		seq_commit(sf, -1);
		return 0;
	}
56
	memset(buf, 0, PAGE_SIZE);
L
Linus Torvalds 已提交
57

58
	/*
59 60
	 * Invoke show().  Control may reach here via seq file lseek even
	 * if @ops->show() isn't implemented.
61
	 */
62
	if (ops->show) {
63
		count = ops->show(kobj, of->kn->priv, buf);
64 65 66
		if (count < 0)
			return count;
	}
67

68 69 70 71
	/*
	 * The code works fine with PAGE_SIZE return but it's likely to
	 * indicate truncated result or overflow in normal use cases.
	 */
72 73 74 75 76 77
	if (count >= (ssize_t)PAGE_SIZE) {
		print_symbol("fill_read_buffer: %s returned bad count\n",
			(unsigned long)ops->show);
		/* Try to struggle along */
		count = PAGE_SIZE - 1;
	}
78 79
	seq_commit(sf, count);
	return 0;
L
Linus Torvalds 已提交
80 81
}

82
static ssize_t sysfs_kf_bin_read(struct kernfs_open_file *of, char *buf,
83
				 size_t count, loff_t pos)
T
Tejun Heo 已提交
84
{
85
	struct bin_attribute *battr = of->kn->priv;
86
	struct kobject *kobj = of->kn->parent->priv;
87
	loff_t size = file_inode(of->file)->i_size;
T
Tejun Heo 已提交
88

89
	if (!count)
T
Tejun Heo 已提交
90 91 92
		return 0;

	if (size) {
93
		if (pos > size)
T
Tejun Heo 已提交
94
			return 0;
95 96
		if (pos + count > size)
			count = size - pos;
T
Tejun Heo 已提交
97 98
	}

99 100 101 102 103 104
	if (!battr->read)
		return -EIO;

	return battr->read(of->file, kobj, battr, buf, pos, count);
}

105
/* kernfs write callback for regular sysfs files */
106
static ssize_t sysfs_kf_write(struct kernfs_open_file *of, char *buf,
107
			      size_t count, loff_t pos)
L
Linus Torvalds 已提交
108
{
109
	const struct sysfs_ops *ops = sysfs_file_ops(of->kn);
110
	struct kobject *kobj = of->kn->parent->priv;
111

112 113
	if (!count)
		return 0;
114

115
	return ops->store(kobj, of->kn->priv, buf, count);
116
}
117

118
/* kernfs write callback for bin sysfs files */
119
static ssize_t sysfs_kf_bin_write(struct kernfs_open_file *of, char *buf,
120 121
				  size_t count, loff_t pos)
{
122
	struct bin_attribute *battr = of->kn->priv;
123
	struct kobject *kobj = of->kn->parent->priv;
124
	loff_t size = file_inode(of->file)->i_size;
125

126 127
	if (size) {
		if (size <= pos)
128
			return -EFBIG;
129
		count = min_t(ssize_t, count, size - pos);
130
	}
131 132
	if (!count)
		return 0;
133

134 135
	if (!battr->write)
		return -EIO;
L
Linus Torvalds 已提交
136

137
	return battr->write(of->file, kobj, battr, buf, pos, count);
L
Linus Torvalds 已提交
138 139
}

140
static int sysfs_kf_bin_mmap(struct kernfs_open_file *of,
141 142
			     struct vm_area_struct *vma)
{
143
	struct bin_attribute *battr = of->kn->priv;
144
	struct kobject *kobj = of->kn->parent->priv;
145 146 147 148

	return battr->mmap(of->file, kobj, battr, vma);
}

149
void sysfs_notify(struct kobject *kobj, const char *dir, const char *attr)
150
{
151
	struct kernfs_node *kn = kobj->sd, *tmp;
152

153 154
	if (kn && dir)
		kn = kernfs_find_and_get(kn, dir);
155
	else
156
		kernfs_get(kn);
157

158 159 160 161
	if (kn && attr) {
		tmp = kernfs_find_and_get(kn, attr);
		kernfs_put(kn);
		kn = tmp;
162
	}
163

164 165 166
	if (kn) {
		kernfs_notify(kn);
		kernfs_put(kn);
167
	}
168 169 170
}
EXPORT_SYMBOL_GPL(sysfs_notify);

T
Tejun Heo 已提交
171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197
static const struct kernfs_ops sysfs_file_kfops_empty = {
};

static const struct kernfs_ops sysfs_file_kfops_ro = {
	.seq_show	= sysfs_kf_seq_show,
};

static const struct kernfs_ops sysfs_file_kfops_wo = {
	.write		= sysfs_kf_write,
};

static const struct kernfs_ops sysfs_file_kfops_rw = {
	.seq_show	= sysfs_kf_seq_show,
	.write		= sysfs_kf_write,
};

static const struct kernfs_ops sysfs_bin_kfops_ro = {
	.read		= sysfs_kf_bin_read,
};

static const struct kernfs_ops sysfs_bin_kfops_wo = {
	.write		= sysfs_kf_bin_write,
};

static const struct kernfs_ops sysfs_bin_kfops_rw = {
	.read		= sysfs_kf_bin_read,
	.write		= sysfs_kf_bin_write,
198 199 200 201 202
};

static const struct kernfs_ops sysfs_bin_kfops_mmap = {
	.read		= sysfs_kf_bin_read,
	.write		= sysfs_kf_bin_write,
T
Tejun Heo 已提交
203 204 205
	.mmap		= sysfs_kf_bin_mmap,
};

206
int sysfs_add_file_mode_ns(struct kernfs_node *parent,
207
			   const struct attribute *attr, bool is_bin,
208
			   umode_t mode, const void *ns)
L
Linus Torvalds 已提交
209
{
210
	struct lock_class_key *key = NULL;
T
Tejun Heo 已提交
211
	const struct kernfs_ops *ops;
212
	struct kernfs_node *kn;
213
	loff_t size;
L
Linus Torvalds 已提交
214

215
	if (!is_bin) {
216
		struct kobject *kobj = parent->priv;
T
Tejun Heo 已提交
217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232
		const struct sysfs_ops *sysfs_ops = kobj->ktype->sysfs_ops;

		/* every kobject with an attribute needs a ktype assigned */
		if (WARN(!sysfs_ops, KERN_ERR
			 "missing sysfs attribute operations for kobject: %s\n",
			 kobject_name(kobj)))
			return -EINVAL;

		if (sysfs_ops->show && sysfs_ops->store)
			ops = &sysfs_file_kfops_rw;
		else if (sysfs_ops->show)
			ops = &sysfs_file_kfops_ro;
		else if (sysfs_ops->store)
			ops = &sysfs_file_kfops_wo;
		else
			ops = &sysfs_file_kfops_empty;
233 234

		size = PAGE_SIZE;
T
Tejun Heo 已提交
235 236 237
	} else {
		struct bin_attribute *battr = (void *)attr;

238 239 240
		if (battr->mmap)
			ops = &sysfs_bin_kfops_mmap;
		else if (battr->read && battr->write)
T
Tejun Heo 已提交
241 242 243 244 245 246 247
			ops = &sysfs_bin_kfops_rw;
		else if (battr->read)
			ops = &sysfs_bin_kfops_ro;
		else if (battr->write)
			ops = &sysfs_bin_kfops_wo;
		else
			ops = &sysfs_file_kfops_empty;
248 249

		size = battr->size;
T
Tejun Heo 已提交
250 251
	}

252 253 254 255
#ifdef CONFIG_DEBUG_LOCK_ALLOC
	if (!attr->ignore_lockdep)
		key = attr->key ?: (struct lock_class_key *)&attr->skey;
#endif
256 257
	kn = __kernfs_create_file(parent, attr->name, mode, size, ops,
				  (void *)attr, ns, true, key);
258 259 260 261
	if (IS_ERR(kn)) {
		if (PTR_ERR(kn) == -EEXIST)
			sysfs_warn_dup(parent, attr->name);
		return PTR_ERR(kn);
262 263 264 265
	}
	return 0;
}

266
int sysfs_add_file(struct kernfs_node *parent, const struct attribute *attr,
267
		   bool is_bin)
268
{
269
	return sysfs_add_file_mode_ns(parent, attr, is_bin, attr->mode, NULL);
270 271
}

L
Linus Torvalds 已提交
272
/**
273 274 275 276
 * sysfs_create_file_ns - create an attribute file for an object with custom ns
 * @kobj: object we're creating for
 * @attr: attribute descriptor
 * @ns: namespace the new file should belong to
L
Linus Torvalds 已提交
277
 */
278 279
int sysfs_create_file_ns(struct kobject *kobj, const struct attribute *attr,
			 const void *ns)
L
Linus Torvalds 已提交
280
{
281
	BUG_ON(!kobj || !kobj->sd || !attr);
L
Linus Torvalds 已提交
282

283
	return sysfs_add_file_mode_ns(kobj->sd, attr, false, attr->mode, ns);
L
Linus Torvalds 已提交
284 285

}
286
EXPORT_SYMBOL_GPL(sysfs_create_file_ns);
L
Linus Torvalds 已提交
287

288 289 290 291 292 293 294 295 296 297 298 299
int sysfs_create_files(struct kobject *kobj, const struct attribute **ptr)
{
	int err = 0;
	int i;

	for (i = 0; ptr[i] && !err; i++)
		err = sysfs_create_file(kobj, ptr[i]);
	if (err)
		while (--i >= 0)
			sysfs_remove_file(kobj, ptr[i]);
	return err;
}
300
EXPORT_SYMBOL_GPL(sysfs_create_files);
L
Linus Torvalds 已提交
301

302 303 304 305 306 307 308 309 310
/**
 * sysfs_add_file_to_group - add an attribute file to a pre-existing group.
 * @kobj: object we're acting for.
 * @attr: attribute descriptor.
 * @group: group name.
 */
int sysfs_add_file_to_group(struct kobject *kobj,
		const struct attribute *attr, const char *group)
{
311
	struct kernfs_node *parent;
312 313
	int error;

314
	if (group) {
315
		parent = kernfs_find_and_get(kobj->sd, group);
316
	} else {
317 318
		parent = kobj->sd;
		kernfs_get(parent);
319
	}
320

321
	if (!parent)
322 323
		return -ENOENT;

324 325
	error = sysfs_add_file(parent, attr, false);
	kernfs_put(parent);
326

327 328 329 330
	return error;
}
EXPORT_SYMBOL_GPL(sysfs_add_file_to_group);

331 332 333 334 335 336 337
/**
 * sysfs_chmod_file - update the modified mode value on an object attribute.
 * @kobj: object we're acting for.
 * @attr: attribute descriptor.
 * @mode: file permissions.
 *
 */
338
int sysfs_chmod_file(struct kobject *kobj, const struct attribute *attr,
A
Al Viro 已提交
339
		     umode_t mode)
340
{
341
	struct kernfs_node *kn;
342
	struct iattr newattrs;
343 344
	int rc;

345 346
	kn = kernfs_find_and_get(kobj->sd, attr->name);
	if (!kn)
347
		return -ENOENT;
348

349
	newattrs.ia_mode = (mode & S_IALLUGO) | (kn->mode & ~S_IALLUGO);
350
	newattrs.ia_valid = ATTR_MODE;
351

352
	rc = kernfs_setattr(kn, &newattrs);
353

354
	kernfs_put(kn);
355
	return rc;
356 357 358
}
EXPORT_SYMBOL_GPL(sysfs_chmod_file);

L
Linus Torvalds 已提交
359
/**
360 361 362 363
 * sysfs_remove_file_ns - remove an object attribute with a custom ns tag
 * @kobj: object we're acting for
 * @attr: attribute descriptor
 * @ns: namespace tag of the file to remove
L
Linus Torvalds 已提交
364
 *
365
 * Hash the attribute name and namespace tag and kill the victim.
L
Linus Torvalds 已提交
366
 */
367 368
void sysfs_remove_file_ns(struct kobject *kobj, const struct attribute *attr,
			  const void *ns)
L
Linus Torvalds 已提交
369
{
370
	struct kernfs_node *parent = kobj->sd;
371

372
	kernfs_remove_by_name_ns(parent, attr->name, ns);
L
Linus Torvalds 已提交
373
}
374
EXPORT_SYMBOL_GPL(sysfs_remove_file_ns);
L
Linus Torvalds 已提交
375

376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398
/**
 * sysfs_remove_file_self - remove an object attribute from its own method
 * @kobj: object we're acting for
 * @attr: attribute descriptor
 *
 * See kernfs_remove_self() for details.
 */
bool sysfs_remove_file_self(struct kobject *kobj, const struct attribute *attr)
{
	struct kernfs_node *parent = kobj->sd;
	struct kernfs_node *kn;
	bool ret;

	kn = kernfs_find_and_get(parent, attr->name);
	if (WARN_ON_ONCE(!kn))
		return false;

	ret = kernfs_remove_self(kn);

	kernfs_put(kn);
	return ret;
}

399
void sysfs_remove_files(struct kobject *kobj, const struct attribute **ptr)
400 401 402 403 404
{
	int i;
	for (i = 0; ptr[i]; i++)
		sysfs_remove_file(kobj, ptr[i]);
}
405
EXPORT_SYMBOL_GPL(sysfs_remove_files);
L
Linus Torvalds 已提交
406

407 408 409 410 411 412 413 414 415
/**
 * sysfs_remove_file_from_group - remove an attribute file from a group.
 * @kobj: object we're acting for.
 * @attr: attribute descriptor.
 * @group: group name.
 */
void sysfs_remove_file_from_group(struct kobject *kobj,
		const struct attribute *attr, const char *group)
{
416
	struct kernfs_node *parent;
417

418
	if (group) {
419
		parent = kernfs_find_and_get(kobj->sd, group);
420
	} else {
421 422
		parent = kobj->sd;
		kernfs_get(parent);
423 424
	}

425 426 427
	if (parent) {
		kernfs_remove_by_name(parent, attr->name);
		kernfs_put(parent);
428 429 430 431
	}
}
EXPORT_SYMBOL_GPL(sysfs_remove_file_from_group);

432 433 434 435 436 437 438 439 440 441
/**
 *	sysfs_create_bin_file - create binary file for object.
 *	@kobj:	object.
 *	@attr:	attribute descriptor.
 */
int sysfs_create_bin_file(struct kobject *kobj,
			  const struct bin_attribute *attr)
{
	BUG_ON(!kobj || !kobj->sd || !attr);

442
	return sysfs_add_file(kobj->sd, &attr->attr, true);
443 444 445 446 447 448 449 450 451 452 453
}
EXPORT_SYMBOL_GPL(sysfs_create_bin_file);

/**
 *	sysfs_remove_bin_file - remove binary file for object.
 *	@kobj:	object.
 *	@attr:	attribute descriptor.
 */
void sysfs_remove_bin_file(struct kobject *kobj,
			   const struct bin_attribute *attr)
{
454
	kernfs_remove_by_name(kobj->sd, attr->attr.name);
455 456
}
EXPORT_SYMBOL_GPL(sysfs_remove_bin_file);