162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Simple file system for zoned block devices exposing zones as files. 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2022 Western Digital Corporation or its affiliates. 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci#include <linux/fs.h> 862306a36Sopenharmony_ci#include <linux/seq_file.h> 962306a36Sopenharmony_ci#include <linux/blkdev.h> 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ci#include "zonefs.h" 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_cistruct zonefs_sysfs_attr { 1462306a36Sopenharmony_ci struct attribute attr; 1562306a36Sopenharmony_ci ssize_t (*show)(struct zonefs_sb_info *sbi, char *buf); 1662306a36Sopenharmony_ci}; 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci#define ZONEFS_SYSFS_ATTR_RO(name) \ 1962306a36Sopenharmony_cistatic struct zonefs_sysfs_attr zonefs_sysfs_attr_##name = __ATTR_RO(name) 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci#define ATTR_LIST(name) &zonefs_sysfs_attr_##name.attr 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_cistatic ssize_t zonefs_sysfs_attr_show(struct kobject *kobj, 2462306a36Sopenharmony_ci struct attribute *attr, char *buf) 2562306a36Sopenharmony_ci{ 2662306a36Sopenharmony_ci struct zonefs_sb_info *sbi = 2762306a36Sopenharmony_ci container_of(kobj, struct zonefs_sb_info, s_kobj); 2862306a36Sopenharmony_ci struct zonefs_sysfs_attr *zonefs_attr = 2962306a36Sopenharmony_ci container_of(attr, struct zonefs_sysfs_attr, attr); 3062306a36Sopenharmony_ci 3162306a36Sopenharmony_ci if (!zonefs_attr->show) 3262306a36Sopenharmony_ci return 0; 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci return zonefs_attr->show(sbi, buf); 3562306a36Sopenharmony_ci} 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_cistatic ssize_t max_wro_seq_files_show(struct zonefs_sb_info *sbi, char *buf) 3862306a36Sopenharmony_ci{ 3962306a36Sopenharmony_ci return sysfs_emit(buf, "%u\n", sbi->s_max_wro_seq_files); 4062306a36Sopenharmony_ci} 4162306a36Sopenharmony_ciZONEFS_SYSFS_ATTR_RO(max_wro_seq_files); 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_cistatic ssize_t nr_wro_seq_files_show(struct zonefs_sb_info *sbi, char *buf) 4462306a36Sopenharmony_ci{ 4562306a36Sopenharmony_ci return sysfs_emit(buf, "%d\n", atomic_read(&sbi->s_wro_seq_files)); 4662306a36Sopenharmony_ci} 4762306a36Sopenharmony_ciZONEFS_SYSFS_ATTR_RO(nr_wro_seq_files); 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_cistatic ssize_t max_active_seq_files_show(struct zonefs_sb_info *sbi, char *buf) 5062306a36Sopenharmony_ci{ 5162306a36Sopenharmony_ci return sysfs_emit(buf, "%u\n", sbi->s_max_active_seq_files); 5262306a36Sopenharmony_ci} 5362306a36Sopenharmony_ciZONEFS_SYSFS_ATTR_RO(max_active_seq_files); 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_cistatic ssize_t nr_active_seq_files_show(struct zonefs_sb_info *sbi, char *buf) 5662306a36Sopenharmony_ci{ 5762306a36Sopenharmony_ci return sysfs_emit(buf, "%d\n", atomic_read(&sbi->s_active_seq_files)); 5862306a36Sopenharmony_ci} 5962306a36Sopenharmony_ciZONEFS_SYSFS_ATTR_RO(nr_active_seq_files); 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_cistatic struct attribute *zonefs_sysfs_attrs[] = { 6262306a36Sopenharmony_ci ATTR_LIST(max_wro_seq_files), 6362306a36Sopenharmony_ci ATTR_LIST(nr_wro_seq_files), 6462306a36Sopenharmony_ci ATTR_LIST(max_active_seq_files), 6562306a36Sopenharmony_ci ATTR_LIST(nr_active_seq_files), 6662306a36Sopenharmony_ci NULL, 6762306a36Sopenharmony_ci}; 6862306a36Sopenharmony_ciATTRIBUTE_GROUPS(zonefs_sysfs); 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_cistatic void zonefs_sysfs_sb_release(struct kobject *kobj) 7162306a36Sopenharmony_ci{ 7262306a36Sopenharmony_ci struct zonefs_sb_info *sbi = 7362306a36Sopenharmony_ci container_of(kobj, struct zonefs_sb_info, s_kobj); 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci complete(&sbi->s_kobj_unregister); 7662306a36Sopenharmony_ci} 7762306a36Sopenharmony_ci 7862306a36Sopenharmony_cistatic const struct sysfs_ops zonefs_sysfs_attr_ops = { 7962306a36Sopenharmony_ci .show = zonefs_sysfs_attr_show, 8062306a36Sopenharmony_ci}; 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_cistatic const struct kobj_type zonefs_sb_ktype = { 8362306a36Sopenharmony_ci .default_groups = zonefs_sysfs_groups, 8462306a36Sopenharmony_ci .sysfs_ops = &zonefs_sysfs_attr_ops, 8562306a36Sopenharmony_ci .release = zonefs_sysfs_sb_release, 8662306a36Sopenharmony_ci}; 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_cistatic struct kobject *zonefs_sysfs_root; 8962306a36Sopenharmony_ci 9062306a36Sopenharmony_ciint zonefs_sysfs_register(struct super_block *sb) 9162306a36Sopenharmony_ci{ 9262306a36Sopenharmony_ci struct zonefs_sb_info *sbi = ZONEFS_SB(sb); 9362306a36Sopenharmony_ci int ret; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci init_completion(&sbi->s_kobj_unregister); 9662306a36Sopenharmony_ci ret = kobject_init_and_add(&sbi->s_kobj, &zonefs_sb_ktype, 9762306a36Sopenharmony_ci zonefs_sysfs_root, "%s", sb->s_id); 9862306a36Sopenharmony_ci if (ret) { 9962306a36Sopenharmony_ci kobject_put(&sbi->s_kobj); 10062306a36Sopenharmony_ci wait_for_completion(&sbi->s_kobj_unregister); 10162306a36Sopenharmony_ci return ret; 10262306a36Sopenharmony_ci } 10362306a36Sopenharmony_ci 10462306a36Sopenharmony_ci sbi->s_sysfs_registered = true; 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_ci return 0; 10762306a36Sopenharmony_ci} 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_civoid zonefs_sysfs_unregister(struct super_block *sb) 11062306a36Sopenharmony_ci{ 11162306a36Sopenharmony_ci struct zonefs_sb_info *sbi = ZONEFS_SB(sb); 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_ci if (!sbi || !sbi->s_sysfs_registered) 11462306a36Sopenharmony_ci return; 11562306a36Sopenharmony_ci 11662306a36Sopenharmony_ci kobject_del(&sbi->s_kobj); 11762306a36Sopenharmony_ci kobject_put(&sbi->s_kobj); 11862306a36Sopenharmony_ci wait_for_completion(&sbi->s_kobj_unregister); 11962306a36Sopenharmony_ci} 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ciint __init zonefs_sysfs_init(void) 12262306a36Sopenharmony_ci{ 12362306a36Sopenharmony_ci zonefs_sysfs_root = kobject_create_and_add("zonefs", fs_kobj); 12462306a36Sopenharmony_ci if (!zonefs_sysfs_root) 12562306a36Sopenharmony_ci return -ENOMEM; 12662306a36Sopenharmony_ci 12762306a36Sopenharmony_ci return 0; 12862306a36Sopenharmony_ci} 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_civoid zonefs_sysfs_exit(void) 13162306a36Sopenharmony_ci{ 13262306a36Sopenharmony_ci kobject_put(zonefs_sysfs_root); 13362306a36Sopenharmony_ci zonefs_sysfs_root = NULL; 13462306a36Sopenharmony_ci} 135