blob: fa3dc66d82f16e85e183fe67e5979ed51d56f83f [file] [log] [blame]
Brian Fostera31b1d32014-07-15 08:07:01 +10001/*
2 * Copyright (c) 2014 Red Hat, Inc.
3 * All Rights Reserved.
4 *
5 * This program is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU General Public License as
7 * published by the Free Software Foundation.
8 *
9 * This program is distributed in the hope that it would be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write the Free Software Foundation,
16 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
17 */
18
19#include "xfs.h"
20#include "xfs_sysfs.h"
Brian Fosterbaff4e42014-07-15 08:07:29 +100021#include "xfs_log_format.h"
22#include "xfs_log.h"
23#include "xfs_log_priv.h"
Bill O'Donnellbb230c12015-10-12 05:15:45 +110024#include "xfs_stats.h"
Brian Fostera31b1d32014-07-15 08:07:01 +100025
26struct xfs_sysfs_attr {
27 struct attribute attr;
Bill O'Donnella27c26402015-10-12 05:18:45 +110028 ssize_t (*show)(struct kobject *kobject, char *buf);
29 ssize_t (*store)(struct kobject *kobject, const char *buf,
30 size_t count);
Brian Fostera31b1d32014-07-15 08:07:01 +100031};
32
33static inline struct xfs_sysfs_attr *
34to_attr(struct attribute *attr)
35{
36 return container_of(attr, struct xfs_sysfs_attr, attr);
37}
38
39#define XFS_SYSFS_ATTR_RW(name) \
40 static struct xfs_sysfs_attr xfs_sysfs_attr_##name = __ATTR_RW(name)
41#define XFS_SYSFS_ATTR_RO(name) \
42 static struct xfs_sysfs_attr xfs_sysfs_attr_##name = __ATTR_RO(name)
Bill O'Donnellbb230c12015-10-12 05:15:45 +110043#define XFS_SYSFS_ATTR_WO(name) \
44 static struct xfs_sysfs_attr xfs_sysfs_attr_##name = __ATTR_WO(name)
Brian Fostera31b1d32014-07-15 08:07:01 +100045
46#define ATTR_LIST(name) &xfs_sysfs_attr_##name.attr
47
48/*
49 * xfs_mount kobject. This currently has no attributes and thus no need for show
50 * and store helpers. The mp kobject serves as the per-mount parent object that
51 * is identified by the fsname under sysfs.
52 */
53
54struct kobj_type xfs_mp_ktype = {
55 .release = xfs_sysfs_release,
56};
Brian Fosterbaff4e42014-07-15 08:07:29 +100057
Bill O'Donnella27c26402015-10-12 05:18:45 +110058STATIC ssize_t
59xfs_sysfs_object_show(
60 struct kobject *kobject,
61 struct attribute *attr,
62 char *buf)
63{
64 struct xfs_sysfs_attr *xfs_attr = to_attr(attr);
65
66 return xfs_attr->show ? xfs_attr->show(kobject, buf) : 0;
67}
68
69STATIC ssize_t
70xfs_sysfs_object_store(
71 struct kobject *kobject,
72 struct attribute *attr,
73 const char *buf,
74 size_t count)
75{
76 struct xfs_sysfs_attr *xfs_attr = to_attr(attr);
77
78 return xfs_attr->store ? xfs_attr->store(kobject, buf, count) : 0;
79}
80
81static const struct sysfs_ops xfs_sysfs_ops = {
82 .show = xfs_sysfs_object_show,
83 .store = xfs_sysfs_object_store,
84};
85
Brian Foster65b65732014-09-09 11:52:42 +100086#ifdef DEBUG
87/* debug */
88
Brian Foster2e227172014-09-09 11:56:13 +100089STATIC ssize_t
90log_recovery_delay_store(
Bill O'Donnella27c26402015-10-12 05:18:45 +110091 struct kobject *kobject,
Brian Foster2e227172014-09-09 11:56:13 +100092 const char *buf,
Bill O'Donnella27c26402015-10-12 05:18:45 +110093 size_t count)
Brian Foster2e227172014-09-09 11:56:13 +100094{
95 int ret;
96 int val;
97
98 ret = kstrtoint(buf, 0, &val);
99 if (ret)
100 return ret;
101
102 if (val < 0 || val > 60)
103 return -EINVAL;
104
105 xfs_globals.log_recovery_delay = val;
106
107 return count;
108}
109
110STATIC ssize_t
111log_recovery_delay_show(
Bill O'Donnella27c26402015-10-12 05:18:45 +1100112 struct kobject *kobject,
113 char *buf)
Brian Foster2e227172014-09-09 11:56:13 +1000114{
115 return snprintf(buf, PAGE_SIZE, "%d\n", xfs_globals.log_recovery_delay);
116}
117XFS_SYSFS_ATTR_RW(log_recovery_delay);
118
Brian Foster65b65732014-09-09 11:52:42 +1000119static struct attribute *xfs_dbg_attrs[] = {
Brian Foster2e227172014-09-09 11:56:13 +1000120 ATTR_LIST(log_recovery_delay),
Brian Foster65b65732014-09-09 11:52:42 +1000121 NULL,
122};
123
Brian Foster65b65732014-09-09 11:52:42 +1000124struct kobj_type xfs_dbg_ktype = {
125 .release = xfs_sysfs_release,
Bill O'Donnella27c26402015-10-12 05:18:45 +1100126 .sysfs_ops = &xfs_sysfs_ops,
Brian Foster65b65732014-09-09 11:52:42 +1000127 .default_attrs = xfs_dbg_attrs,
128};
129
130#endif /* DEBUG */
131
Bill O'Donnellbb230c12015-10-12 05:15:45 +1100132/* stats */
133
134STATIC ssize_t
135stats_show(
Bill O'Donnella27c26402015-10-12 05:18:45 +1100136 struct kobject *kobject,
137 char *buf)
Bill O'Donnellbb230c12015-10-12 05:15:45 +1100138{
139 return xfs_stats_format(buf);
140}
141XFS_SYSFS_ATTR_RO(stats);
142
143STATIC ssize_t
144stats_clear_store(
Bill O'Donnella27c26402015-10-12 05:18:45 +1100145 struct kobject *kobject,
Bill O'Donnellbb230c12015-10-12 05:15:45 +1100146 const char *buf,
Bill O'Donnella27c26402015-10-12 05:18:45 +1100147 size_t count)
Bill O'Donnellbb230c12015-10-12 05:15:45 +1100148{
149 int ret;
150 int val;
151
152 ret = kstrtoint(buf, 0, &val);
153 if (ret)
154 return ret;
155
156 if (val != 1)
157 return -EINVAL;
158 xfs_stats_clearall();
159 return count;
160}
161XFS_SYSFS_ATTR_WO(stats_clear);
162
163static struct attribute *xfs_stats_attrs[] = {
164 ATTR_LIST(stats),
165 ATTR_LIST(stats_clear),
166 NULL,
167};
168
Bill O'Donnellbb230c12015-10-12 05:15:45 +1100169struct kobj_type xfs_stats_ktype = {
170 .release = xfs_sysfs_release,
Bill O'Donnella27c26402015-10-12 05:18:45 +1100171 .sysfs_ops = &xfs_sysfs_ops,
Bill O'Donnellbb230c12015-10-12 05:15:45 +1100172 .default_attrs = xfs_stats_attrs,
173};
174
Brian Fosterbaff4e42014-07-15 08:07:29 +1000175/* xlog */
176
Bill O'Donnella27c26402015-10-12 05:18:45 +1100177static inline struct xlog *
178to_xlog(struct kobject *kobject)
179{
180 struct xfs_kobj *kobj = to_kobj(kobject);
181
182 return container_of(kobj, struct xlog, l_kobj);
183}
184
Brian Foster80d6d692014-07-15 08:07:48 +1000185STATIC ssize_t
186log_head_lsn_show(
Bill O'Donnella27c26402015-10-12 05:18:45 +1100187 struct kobject *kobject,
188 char *buf)
Brian Foster80d6d692014-07-15 08:07:48 +1000189{
Brian Foster80d6d692014-07-15 08:07:48 +1000190 int cycle;
191 int block;
Bill O'Donnella27c26402015-10-12 05:18:45 +1100192 struct xlog *log = to_xlog(kobject);
Brian Foster80d6d692014-07-15 08:07:48 +1000193
194 spin_lock(&log->l_icloglock);
195 cycle = log->l_curr_cycle;
196 block = log->l_curr_block;
197 spin_unlock(&log->l_icloglock);
198
199 return snprintf(buf, PAGE_SIZE, "%d:%d\n", cycle, block);
200}
201XFS_SYSFS_ATTR_RO(log_head_lsn);
202
203STATIC ssize_t
204log_tail_lsn_show(
Bill O'Donnella27c26402015-10-12 05:18:45 +1100205 struct kobject *kobject,
206 char *buf)
Brian Foster80d6d692014-07-15 08:07:48 +1000207{
Brian Foster80d6d692014-07-15 08:07:48 +1000208 int cycle;
209 int block;
Bill O'Donnella27c26402015-10-12 05:18:45 +1100210 struct xlog *log = to_xlog(kobject);
Brian Foster80d6d692014-07-15 08:07:48 +1000211
212 xlog_crack_atomic_lsn(&log->l_tail_lsn, &cycle, &block);
213 return snprintf(buf, PAGE_SIZE, "%d:%d\n", cycle, block);
214}
215XFS_SYSFS_ATTR_RO(log_tail_lsn);
216
217STATIC ssize_t
218reserve_grant_head_show(
Bill O'Donnella27c26402015-10-12 05:18:45 +1100219 struct kobject *kobject,
220 char *buf)
221
Brian Foster80d6d692014-07-15 08:07:48 +1000222{
Brian Foster80d6d692014-07-15 08:07:48 +1000223 int cycle;
224 int bytes;
Bill O'Donnella27c26402015-10-12 05:18:45 +1100225 struct xlog *log = to_xlog(kobject);
Brian Foster80d6d692014-07-15 08:07:48 +1000226
227 xlog_crack_grant_head(&log->l_reserve_head.grant, &cycle, &bytes);
228 return snprintf(buf, PAGE_SIZE, "%d:%d\n", cycle, bytes);
229}
230XFS_SYSFS_ATTR_RO(reserve_grant_head);
231
232STATIC ssize_t
233write_grant_head_show(
Bill O'Donnella27c26402015-10-12 05:18:45 +1100234 struct kobject *kobject,
235 char *buf)
Brian Foster80d6d692014-07-15 08:07:48 +1000236{
Brian Foster80d6d692014-07-15 08:07:48 +1000237 int cycle;
238 int bytes;
Bill O'Donnella27c26402015-10-12 05:18:45 +1100239 struct xlog *log = to_xlog(kobject);
Brian Foster80d6d692014-07-15 08:07:48 +1000240
241 xlog_crack_grant_head(&log->l_write_head.grant, &cycle, &bytes);
242 return snprintf(buf, PAGE_SIZE, "%d:%d\n", cycle, bytes);
243}
244XFS_SYSFS_ATTR_RO(write_grant_head);
245
Brian Fosterbaff4e42014-07-15 08:07:29 +1000246static struct attribute *xfs_log_attrs[] = {
Brian Foster80d6d692014-07-15 08:07:48 +1000247 ATTR_LIST(log_head_lsn),
248 ATTR_LIST(log_tail_lsn),
249 ATTR_LIST(reserve_grant_head),
250 ATTR_LIST(write_grant_head),
Brian Fosterbaff4e42014-07-15 08:07:29 +1000251 NULL,
252};
253
Brian Fosterbaff4e42014-07-15 08:07:29 +1000254struct kobj_type xfs_log_ktype = {
255 .release = xfs_sysfs_release,
Bill O'Donnella27c26402015-10-12 05:18:45 +1100256 .sysfs_ops = &xfs_sysfs_ops,
Brian Fosterbaff4e42014-07-15 08:07:29 +1000257 .default_attrs = xfs_log_attrs,
258};