当前位置: 首页 > article >正文

nfs 中 lockd 与 statd 的作用

lockd 是内核线程,statd 是用户态进程,二者用于 nfs v3 通信协议中,锁状态的恢复(v4 版本协议是有状态协议,无需依赖 statd来恢复状态)。

statd 的监听

当客户端第一次触发 lock 时 (nfs-utils 中可以通过下面的简单c脚本来触发拿锁),

#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <getopt.h>
#include <fcntl.h>

int main() {
    int fd = open(fname, O_RDWR, 0644));
    struct flock    fl;
    fl.l_type = F_RDLCK;
    fl.l_whence = 0;
    fl.l_start = 0;
    fl.l_len = 20;
    if (fcntl(fd, cmd, &fl) < 0) {
        printf("fcntl: failed: %d", errno);
    }
    printf("fcntl: ok\n");
    return 0;
}

内核经过nsm_monitor调用向本地 statd 进程发起 monitor 请求:

nfs3_proc_lock
  ->nlmclnt_proc
  ->nlmclnt_lock
  ->nsm_monitor

nsm_monitor 函数会向本地 rpc.statd 进程发起 NSMPROC_MON 请求记录服务端监听信息。之后调在 nlmclnt_call 函数,向服务端 lockd 线程发起 op_lock 请求。

服务端 lockd 解析参数发现之前没有监听过这个客户端(调用栈如下),就会向服务端本地 rpc.statd 发起 NSMPROC_MON 请求记录客户端监听信息。

nlmsvc_proc_lock
  ->__nlmsvc_proc_lock
  ->nlmsvc_retrieve_args
  ->nsm_monitor 

rpc.statd 监听的客户端会以文件的形式放在 /var/lib/nfs.sm 目录中,至此彼此建立好了监听。

状态的恢复

有两种情况会导致 sm-notify 通知对端自己重启过:

1、rpc.statd 每次启动时会触发一次 sm-notify 进程,但只有第一次触发时会创建 /run/sm-notify.pid 文件,以后检查发现这个文件存在,就不会通知对端。也就是说服务端第一次启动时(service nfs start 时)和客户端第一次启动时(mount -t nfs -o vers=3),rpc.statd 发起的 sm-notify 会检查 /var/lib/nfs.sm 目录中的文件记录,并向其中记录的对端发起 NSMPROC_NOTIFY 请求。

2、但除此之外更一般的情况是机器重启,systemd 系统进程会启动所有注册的 target。其中 /usr/lib/systemd/system/nfs-client.target 会触发 sm-notify 进程向对端发起 NSMPROC_NOTIFY 请求。

rpc.statd 进程收到对端的 NSMPROC_NOTIFY 请求后,会将这一请求加入 notify 列表,并在主循环中将 rpc 请求转发给 lockd 内核线程。调用栈如下:

my_svc_run
  ->process_reply
  ->process_entry 
  ->nsm_xmit_nlmcall

内核收到请求后,调用 nlmsvc_proc_sm_notify->nlm_host_rebooted 恢复或释放锁状态。

  • 服务端处理客户端reboot事件:nlmsvc_free_host_resources 释放 reboot 的客户端申请过的锁
  • 客户端处理服务端reboot事件:nlmclnt_recovery 重新向服务端申请已经申请过的锁。

http://www.kler.cn/news/359492.html

相关文章:

  • 终于找到了!Ubuntu 虚拟机卡死的终极解决方案(如何解决VMware安装Ubuntu24.04桌面进不去、黑屏、虚拟机卡死、显示不全、屏幕闪烁等问题)
  • 10月20日
  • Docker报错-Docker Desktop - Unexpected WSL error
  • [while循环]k的幂
  • 如何利用CMMI帮助组织消除低价值流程
  • 深入理解 Netty 中的 Unpooled 使用方法
  • LeetCode Hot100 | Day4 | 层序遍历有序数组转搜索树验证搜索树搜索树中第K小的元素
  • 如何查看python安装了哪些包
  • 【设计模式】Python 设计模式之建造者模式(Builder Pattern)详解
  • docker网络连接模式详解
  • 基于Android Studio购物商城app+web端,登录实现(前后端分离)二
  • ant design vue TimePicker时间选择器不点击确认也可以设置值
  • 破局汽车基础软件发展丨昂辉科技亮相2024芜湖新能源汽车零部件和后市场生态博览会
  • 【C++STL】list的基本介绍与使用方式
  • Django学习-静态文件
  • 【机器学习】简单易懂的聚类算法K-Means
  • 每日回顾:简单用C写 选择排序、堆排序
  • 基于Android Studio购物商城app+web端实现(前后端分离)一
  • Thread类的基本用用法
  • 基于Multisim旗升降自动控制系统电路(含仿真和报告)