VB.net 2010 视频教程 VB.net 2010 视频教程 python基础视频教程
SQL Server 2008 视频教程 c#入门经典教程 Visual Basic从门到精通视频教程
当前位置:
首页 > go语言 >
  • 系统调用跟踪——ls功能实现(二)

  在上篇文章中我们跟踪ls命令看到了其所使用的这么几个系统调用:stat、openat、fstat、getdents、close、write等,这里再简单介绍下这几个系统调用的功能:
  stat:为获取文件状态系统调用
  openat:将打开目录/data获取它的文件描述符,返回值3即为文件描述符;
  fstat:获取文件描述符为3的文件状态
  getdents64:获取文件描述符为3的目录项
  close:关闭文件描述符3
  write:在获取到文件目录信息后将参数2中的信息标准输出到控制台

系统调用

  在golang中已经给我们封装好了系统调用的相关函数,只要调用相关函数传入所对应的参数即可,go中的系统调用相关函数都放在syscall包下,如在调用sys_getcwd,获取当前工作目录的绝对路径:

//接收目录字节数组
dir:=make([]byte,100)
//调用sys_getcwd系统调用
n,e:= syscall.Getcwd(dir)
fmt.Println(string(dir))

实现ls

  在Linux中将所有的设备都当做文件来处理,用文件描述符来标识每个文件对象;标准输入、标准输出、标准错误的文件描述符分别为:0、1、2;
ls指令的具体实现流程为:获取目录状态、获取目录项、获取文件状态、获取用户信息;
  在调用getdents64系统调用获取目录项时需要传入该目录的文件描述符,所以需提前获取该目录的文件描述符;

具体的实现流程为:

状态对象stat

st_ino   与该文件关联的inode
st_dev   保存文件的设备
st_uid   文件属主的UID号
st_gid   文件属主的GID号
st_atime 文件上一次被访问的时间
st_ctime 文件的权限、属主、组或内容上一次被修改的时间
st_mtime 文件的内容上一次被修改的时间。(和st_ctime的不同之处显而易见)
st_nlink  该文件上硬连接的个数

dir目录对象

d_ino  int64  //索引节点
off_t  int64  //目录中文件偏移
reclen int16  //长度
d_type int8   //文件类型
d_name []byte //文件名

用户信息: 所属用户、所属组、
权限信息: 目录、文件、符号链接、读写执行

实现伪代码:

//获取目录文件描述符  只读|非阻塞|打开目录|执行系统调用时关闭文件描述符(自动关闭) 
fd,e := syscall.Openat(-0x64,path,syscall.O_RDONLY|syscall.O_NONBLOCK|sysc all.O_DIRECTORY|syscall.O_CLOEXEC, 0666)   
//获取目录状态
e := syscall.Lstat(path, dirStat)
var b = make([]byte, dirStat.Size)
//获得目录项
n, e := syscall.Getdents(fd, b)
buf := bytes.NewBuffer(b)
//目录信息
dir := &dirent{}
var c = 0
//遍历获取目录信息
for ; c < n; {
  _ = binary.Read(buf, binary.LittleEndian, &dir.d_ino)
  binary.Read(buf, binary.LittleEndian, &dir.off_t)
  binary.Read(buf, binary.LittleEndian, &dir.reclen)
  binary.Read(buf, binary.LittleEndian, &dir.d_type)
  //名称长度
  nLen := dir.reclen - 19
  name := buf.Next(int(nLen))
  dir.d_name = name[:len(name)-1]
  //获取目录中文件状态信息
  e := syscall.Stat(filePath, stat)
  if e != nil {
     fmt.Println(e.Error())
  }else {
    mode := stat.Mode
    m := parserAuth(mode)
    //使用uid获取用户名
    u, e := user.LookupId(strconv.Itoa(int(stat.Uid)))
    if e == nil {
      //使用gid获取组名
      g, _ := user.LookupGroupId(u.Gid) //mtim 最后一次修改文件时间,atim 最  后一次访问文件时间 ctim最后一次改变文件状态时间
      opt:=&option{m,u.Username,g.Name,int(stat.Size),time.Unix(stat.Mtim.Unix()).Format(TIME_LAYOUT), string(dir.d_name)}
      list = append(list, opt)
     }
   }
   c = c + int(dir.reclen)
 }

程序执行:

文章首发地址:https://mp.weixin.qq.com/s/Xn_xUHei10sgWkqEhMZo5g


相关教程