exfat文件系统无法NFS导出的问题

2024-06-16 23:12

本文主要是介绍exfat文件系统无法NFS导出的问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        最近项目中移植了exfat-linux驱动,但发现exfat格式的U盘无法用exportfs命令在NFS上导出。这篇文章记录了分析、解决方法。

一、问题现象

问题描述:exfat驱动更新后,exfat格式的U盘用exportfs命令NFS导出会报错

$ exportfs -o ro,fsid=0,no_root_squash 192.168.100.74:/storage/9CE5-DFCB/Carlog

报错:"does not support NFS export"


问题原因:

        文件系统要被export必须满足两个条件:第一要有唯一标识,可以是fsid,uuid或者是设备号,第二是文件系统必须设置了export操作集 export_operations。第一个条件可以通过exportfs或在/etc/exports文件中指定fsid的值来满足,第二个条件需要文件系统源代码定义了export_operations

        在exfat-linux驱动源码 fs/exfat/super.c 未定义 export_operations,因此所以exfat文件系统导出NFS会报错。

二、分析过程及思考

1、exportfs命令介绍

        exportfs是Linux中的一个命令,用于将指定的目录或文件系统在NFS上导出,从而实现文件系统的共享和访问控制,通过NFS协议允许其他计算机通过网络访问共享文件。

具体来说,exportfs的作用包括:

  • 共享文件系统:通过exportfs命令,可以将文件系统或目录共享给其他计算机。这样,其他计算机就可以通过NFS协议挂载该共享文件系统,实现文件的共享和访问。
  • 访问权限控制:exportfs命令可以设置文件系统的访问权限,包括读写权限和只读权限。通过配置exportfs,可以控制哪些计算机可以访问共享文件系统以及访问权限的级别。
  • 配置NFS服务:exportfs命令是NFS服务的一部分,通过该命令可以将指定的目录或文件系统添加到NFS服务的导出列表中,从而启用NFS服务并提供共享。

2、分析过程

        首先看下 linux/include/linux/exportfs.h 文档,其中提到 export_operations,详细使用 export_operations 方法需阅读 Documentation/filesystems/nfs/exporting.rst 文档。

      linux/Documentation/filesystems/nfs/exporting.rst 文档,其中有一段:

A file system implementation declares that instances of the filesystem
are exportable by setting the s_export_op field in the struct
super_block.  This field must point to a "struct export_operations"
struct which has the following members:encode_fh  (optional)Takes a dentry and creates a filehandle fragment which can later be usedto find or create a dentry for the same object.  The defaultimplementation creates a filehandle fragment that encodes a 32bit inodeand generation number for the inode encoded, and if necessary thesame information for the parent.fh_to_dentry (mandatory)Given a filehandle fragment, this should find the implied object andcreate a dentry for it (possibly with d_obtain_alias).fh_to_parent (optional but strongly recommended)Given a filehandle fragment, this should find the parent of theimplied object and create a dentry for it (possibly withd_obtain_alias).  May fail if the filehandle fragment is too small.

        文件系统导出功能可通过设置 super_block 中的 s_export_op 字段实现。s_export_op 字段必须指向 "struct export_operations"。

        可以看到,exfat-linux驱动源码 fs/exfat/super.c 并未定义 export_operations,因此NFS导出会失败。

3、解决方法

        将老exfat驱动中 s_export_op 相关代码移植到 exfat-linux驱动源码 fs/exfat/super.c 中,方法如下:

 添加 linux/exportfs.h 头文件

#include <linux/exportfs.h>

定义 struct export_operations 结构体

/*======================================================================*/
/*  Export Operations                                                   */
/*======================================================================*/static struct inode *exfat_nfs_get_inode(struct super_block *sb,u64 ino, u32 generation)
{struct inode *inode = NULL;if (ino < EXFAT_ROOT_INO)return inode;inode = ilookup(sb, ino);if (inode && generation && (inode->i_generation != generation)) {iput(inode);inode = NULL;}return inode;
}static struct dentry *exfat_fh_to_dentry(struct super_block *sb, struct fid *fid,int fh_len, int fh_type)
{return generic_fh_to_dentry(sb, fid, fh_len, fh_type,exfat_nfs_get_inode);
}static struct dentry *exfat_fh_to_parent(struct super_block *sb, struct fid *fid,int fh_len, int fh_type)
{return generic_fh_to_parent(sb, fid, fh_len, fh_type,exfat_nfs_get_inode);
}const struct export_operations exfat_export_ops = {.fh_to_dentry   = exfat_fh_to_dentry,.fh_to_parent   = exfat_fh_to_parent,
};

         s_export_op 指向 struct export_operations 结构体。exfat_fill_super() 函数中添加 sb->s_export_op = &exfat_export_ops;

static int exfat_fill_super(struct super_block *sb, void *data, int silent)
{struct exfat_sb_info *sbi;struct exfat_mount_options *opts;struct inode *root_inode;int err;err = exfat_init_sb_info(sb);if (err) {exfat_err(sb, "failed to initialize superblock info");goto failed;}sbi = sb->s_fs_info;opts = &sbi->options;sb->s_flags |= SB_NODIRATIME;sb->s_magic = EXFAT_SUPER_MAGIC;sb->s_op = &exfat_sops;sb->s_export_op = &exfat_export_ops;。。。 。。。
}

代码下载链接:https://download.csdn.net/download/hinewcc/89440534

说明:相关修改请看super.c文件!!!

修改后,重新编译kernel 运行 NFS 导出功能正常!

4、参考文档

Kernel File Handle 详解 - liuchao719 - 博客园 (cnblogs.com)

【linux3.10】【nfs】使文件系统可导出_linuxnfs导出文件-CSDN博客

jffs2文件系统不支持export的问题_sonfs-CSDN博客

这篇关于exfat文件系统无法NFS导出的问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1067819

相关文章

如何解决mmcv无法安装或安装之后报错问题

《如何解决mmcv无法安装或安装之后报错问题》:本文主要介绍如何解决mmcv无法安装或安装之后报错问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mmcv无法安装或安装之后报错问题1.当我们运行YOwww.chinasem.cnLO时遇到2.找到下图所示这里3.

浅谈配置MMCV环境,解决报错,版本不匹配问题

《浅谈配置MMCV环境,解决报错,版本不匹配问题》:本文主要介绍浅谈配置MMCV环境,解决报错,版本不匹配问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录配置MMCV环境,解决报错,版本不匹配错误示例正确示例总结配置MMCV环境,解决报错,版本不匹配在col

Vue3使用router,params传参为空问题

《Vue3使用router,params传参为空问题》:本文主要介绍Vue3使用router,params传参为空问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录vue3使用China编程router,params传参为空1.使用query方式传参2.使用 Histo

SpringBoot首笔交易慢问题排查与优化方案

《SpringBoot首笔交易慢问题排查与优化方案》在我们的微服务项目中,遇到这样的问题:应用启动后,第一笔交易响应耗时高达4、5秒,而后续请求均能在毫秒级完成,这不仅触发监控告警,也极大影响了用户体... 目录问题背景排查步骤1. 日志分析2. 性能工具定位优化方案:提前预热各种资源1. Flowable

Python将博客内容html导出为Markdown格式

《Python将博客内容html导出为Markdown格式》Python将博客内容html导出为Markdown格式,通过博客url地址抓取文章,分析并提取出文章标题和内容,将内容构建成html,再转... 目录一、为什么要搞?二、准备如何搞?三、说搞咱就搞!抓取文章提取内容构建html转存markdown

vue使用docxtemplater导出word

《vue使用docxtemplater导出word》docxtemplater是一种邮件合并工具,以编程方式使用并处理条件、循环,并且可以扩展以插入任何内容,下面我们来看看如何使用docxtempl... 目录docxtemplatervue使用docxtemplater导出word安装常用语法 封装导出方

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

java中使用POI生成Excel并导出过程

《java中使用POI生成Excel并导出过程》:本文主要介绍java中使用POI生成Excel并导出过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录需求说明及实现方式需求完成通用代码版本1版本2结果展示type参数为atype参数为b总结注:本文章中代码均为

SpringBoot启动报错的11个高频问题排查与解决终极指南

《SpringBoot启动报错的11个高频问题排查与解决终极指南》这篇文章主要为大家详细介绍了SpringBoot启动报错的11个高频问题的排查与解决,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一... 目录1. 依赖冲突:NoSuchMethodError 的终极解法2. Bean注入失败:No qu

MySQL新增字段后Java实体未更新的潜在问题与解决方案

《MySQL新增字段后Java实体未更新的潜在问题与解决方案》在Java+MySQL的开发中,我们通常使用ORM框架来映射数据库表与Java对象,但有时候,数据库表结构变更(如新增字段)后,开发人员可... 目录引言1. 问题背景:数据库与 Java 实体不同步1.1 常见场景1.2 示例代码2. 不同操作