首页 > 其他分享 >【XV6】 Xv6 and Unix utilities

【XV6】 Xv6 and Unix utilities

时间:2024-02-14 19:14:03浏览次数:43  
标签:p2 p1 int utilities Xv6 Unix sleep close sizeof

代码:https://github.com/JasenChao/xv6-labs.git

运行xv6

实验环境使用的是Ubuntu 20.04,需要安装一些工具:

sudo apt-get install git build-essential gdb-multiarch qemu-system-misc gcc-riscv64-linux-gnu binutils-riscv64-linux-gnu

安装完成后验证qemu是否安装成功,成功的话应该打印qemu的版本:

qemu-system-riscv64 --version

同时还需要有RISC-V版本的GCC,以下三个至少有一个正常安装,可以打印版本:

riscv64-linux-gnu-gcc --version
riscv64-unknown-elf-gcc --version
riscv64-unknown-linux-gnu-gcc --version

下载xv6源码并编译运行

git clone git://g.csail.mit.edu/xv6-labs-2023
cd xv6-labs-2023
make qemu

一切正常的话会显示xv6 kernel is booting,以及一个可以输入命令的终端。

实现sleep命令

题目要求实现一个sleep命令,可以使系统睡眠指定的时间。

user文件夹下新建sleep.c文件,要求包含判断参数是否正确:

#include "kernel/types.h"
#include "user/user.h"

int
main(int argc, char *argv[])
{
  if(argc != 2){
    fprintf(2, "usage: sleep times\n");
    exit(1);
  }

  sleep(atoi(argv[1]));

  exit(0);
}

修改Makefile中的以下内容,使得sleep出现在命令列表中:

UPROGS=\
	$U/_cat\
	$U/_echo\
	$U/_forktest\
	$U/_grep\
	$U/_init\
	$U/_kill\
	$U/_ln\
	$U/_ls\
	$U/_mkdir\
	$U/_rm\
	$U/_sh\
	$U/_sleep\
	$U/_stressfs\
	$U/_usertests\
	$U/_grind\
	$U/_wc\
	$U/_zombie\

使用make GRADEFLAGS=sleep grade命令测试是否正确。

实现pingpong命令

题目要求实现一个pingpong命令,父进程传给子进程一个字符,子进程读到这个字符就输出<pid>: received ping,再把这个字符传回给父进程,父进程读到这个字符就输出<pid>: received pong

user文件夹下新建pingpong.c文件:

#include "kernel/types.h"
#include "user/user.h"

int
main(int argc, char *argv[])
{
  int p1[2], p2[2];

  pipe(p1);
  pipe(p2);

  if(fork() == 0){
    char byte;

    close(p1[1]);   // 子进程不需要p1的写端
    close(p2[0]);   // 子进程不需要p2的读端
    
    read(p1[0], &byte, sizeof(byte));
    printf("%d: received ping\n", getpid());

    write(p2[1], &byte, sizeof(byte));

    close(p1[0]);
    close(p2[1]);

    exit(0);
  }else{
    char byte = 's';

    close(p1[0]);   // 父进程不需要p1的读端
    close(p2[1]);   // 父进程不需要p2的写端
    
    write(p1[1], &byte, sizeof(byte));

    read(p2[0], &byte, sizeof(byte));
    printf("%d: received pong\n", getpid());

    close(p1[1]);
    close(p2[0]);

    exit(0);
  }
}

修改Makefile中的以下内容,使得pingpong出现在命令列表中:

UPROGS=\
	$U/_cat\
	$U/_echo\
	$U/_forktest\
	$U/_grep\
	$U/_init\
	$U/_kill\
	$U/_ln\
	$U/_ls\
	$U/_mkdir\
	$U/_pingpong\
	$U/_rm\
	$U/_sh\
	$U/_sleep\
	$U/_stressfs\
	$U/_usertests\
	$U/_grind\
	$U/_wc\
	$U/_zombie\

使用make GRADEFLAGS=pingpong grade命令测试是否正确。

实现primes命令

题目要求实现一个primes命令,筛选2到35之间的所有质数。

user文件夹下新建primes.c文件,其实就是不断用pipe对范围内的数字进行筛选,正常可以用递归的思维不断fork,但是题目中提到xv6资源很少,也不知道是有多少,就用了3个子进程筛选,剩下的用暴力判断的方法筛选:

#include "kernel/types.h"
#include "user/user.h"

int main() {
    int p1[2];
    pipe(p1);

    if (fork()) {
        // 父进程
        close(p1[0]);

        for (int i = 2; i < 36; ++i) {
            write(p1[1], &i, sizeof i);
        }

        close(p1[1]);
    } else {
        // 子进程
        close(p1[1]);

        int base;
        if (read(p1[0], &base, sizeof base) == 0) exit(1);
        printf("prime %d\n", base);

        int p2[2];
        pipe(p2);

        if (fork()) {
            // 还是子进程
            close(p2[0]);
            
            int n;
            while (read(p1[0], &n, sizeof n)) {
                if (n % base != 0) write(p2[1], &n, sizeof n);
            }

            close(p2[1]);
        } else {
            // 孙进程
            close(p2[1]);

            if (read(p2[0], &base, sizeof base) == 0) exit(1);
            printf("prime %d\n", base);

            int p3[2];
            pipe(p3);

            if (fork()) {
                // 还是孙进程
                close(p3[0]);

                int n;
                while (read(p2[0], &n, sizeof n)) {
                    if (n % base != 0) write(p3[1], &n, sizeof n);
                }

                close(p3[1]);
            } else {
                // 曾孙进程
                close(p3[1]);

                while (read(p3[0], &base, sizeof base)) {
                    int flag = 0;
                    for (int i = 2; i * i <= base; ++i) {
                        if (base % i == 0) {
                            flag = 1;
                            break;
                        }
                    }
                    if (flag == 0) printf("prime %d\n", base);
                }
            }
        }

        close(p1[0]);
    }

    wait(0);

    return 0;
}

修改Makefile中的以下内容,使得primes出现在命令列表中:

UPROGS=\
	$U/_cat\
	$U/_echo\
	$U/_find\
	$U/_forktest\
	$U/_grep\
	$U/_init\
	$U/_kill\
	$U/_ln\
	$U/_ls\
	$U/_mkdir\
	$U/_pingpong\
	$U/_primes\
	$U/_rm\
	$U/_sh\
	$U/_sleep\
	$U/_stressfs\
	$U/_usertests\
	$U/_grind\
	$U/_wc\
	$U/_zombie\

使用make GRADEFLAGS=primes grade命令测试是否正确。

实现xargs命令

题目要求实现一个xargs命令,效果相当于Unix中的xargs -n 1

user文件夹下新建xargs.c文件:

#include "kernel/types.h"
#include "user/user.h"

int
main(int argc, char *argv[])
{
    char *cmd;                                                      // 记录xargs后面的命令
    char **new_argv;                                                // 记录xargs后面的命令需要的参数

    cmd = argv[1];                                                  // xargs后面的第一个参数就是命令
    new_argv = malloc(argc * sizeof(char *));                       // 申请argc个指针大小的内存
    for (int i = 0; i < argc - 1; ++i) new_argv[i] = argv[i + 1];   // 将除了xargs以外的内容都保存下来

    char buf[512];                                                  // 用来保存xargs前面的指令输出的内容
    char *pos = buf;                                                // 填充buf的下标指针
    while (read(0, pos, sizeof(char)) > 0) {                        // 每次读一个char,直到没有输出,也就是前面命令的输出内容全部读完了
        if (*pos == '\n' || *pos == '\0') {                         // 如果读出来是换行或者结束符,意味着一条参数读完了,将其统一修改为结束符
            *pos = '\0';
            if (fork() == 0) {                                      // 将最后一个参数补齐,由子进程来完成
                new_argv[argc - 1] = buf;
                exec(cmd, new_argv);
            } else {                                                // 父进程的buf清空,pos归位,准备再读下一条参数
                memset(buf, 0, 512);
                pos = buf;
            }
        } else pos++;                                               // 读到换行和结束符以外的内容就正常写入buf
    }
    wait(0);                                                        // 等待子进程全部结束
    free(new_argv);                                                 // 释放前面malloc的内存

    return 0;
}

修改Makefile中的以下内容,使得xargs出现在命令列表中:

UPROGS=\
	$U/_cat\
	$U/_echo\
	$U/_find\
	$U/_forktest\
	$U/_grep\
	$U/_init\
	$U/_kill\
	$U/_ln\
	$U/_ls\
	$U/_mkdir\
	$U/_pingpong\
	$U/_primes\
	$U/_rm\
	$U/_sh\
	$U/_sleep\
	$U/_stressfs\
	$U/_usertests\
	$U/_grind\
	$U/_wc\
	$U/_xargs\
	$U/_zombie\

使用make GRADEFLAGS=xargs grade命令测试是否正确。

测试结果

使用make grade命令对所有内容进行测试,输出测试结果:

== Test sleep, no arguments ==
$ make qemu-gdb
sleep, no arguments: OK (1.9s)
== Test sleep, returns ==
$ make qemu-gdb
sleep, returns: OK (0.8s)
== Test sleep, makes syscall ==
$ make qemu-gdb
sleep, makes syscall: OK (1.0s)
== Test pingpong ==
$ make qemu-gdb
pingpong: OK (1.0s)
== Test primes ==
$ make qemu-gdb
primes: OK (1.1s)
== Test find, in current directory ==
$ make qemu-gdb
find, in current directory: OK (1.0s)
== Test find, recursive ==
$ make qemu-gdb
find, recursive: OK (1.1s)
== Test xargs ==
$ make qemu-gdb
xargs: OK (1.1s)

标签:p2,p1,int,utilities,Xv6,Unix,sleep,close,sizeof
From: https://www.cnblogs.com/JasenChao/p/18015444

相关文章

  • 【XV6】 page tables
    代码:https://github.com/JasenChao/xv6-labs.git快速获取pid-ugetpid题目要求参考已实现的ugetpid()使用USYSCALL快速获取pid。实现的思路是在每一个进程中增加一个共享页面,通过USYSCALL指定的虚拟地址,找到指定的页面。参考进程中的Trampoline页和Trapframe页。Trampoline页保......
  • 【XV6】 system calls
    代码:https://github.com/JasenChao/xv6-labs.git使用GDB调试安装risc-v的GDB先安装依赖:sudoapt-getinstalllibncurses5-devpython2python2-devtexinfolibreadline-dev再下载源码,可以从清华镜像源下载:wgethttps://mirrors.tuna.tsinghua.edu.cn/gnu/gdb/gdb-13.2.ta......
  • 【XV6】 Multithreading
    代码:https://github.com/JasenChao/xv6-labs.git用户级线程切换题目要求完成用户级线程系统,提示程序要在uthread.c和uthread_switch.S中补充完成。用户级线程调度和进程的机制是类似的,因此uthread_switch.S可以复制swtch.S中的内容: .globlthread_switchthread_switch:......
  • 【XV6】 Copy-on-Write Fork for xv6
    代码:https://github.com/JasenChao/xv6-labs.gitCopy-on-WriteFork系统调用fork()会复制一个父进程的用户空间到子进程,一方面如果进程较大,复制需要很长的时间,另一方面复制的内存的大部分会被丢弃,造成浪费。题目要求实现写时复制COW来延迟fork的物理内存复制,子进程只创建了一个......
  • 【XV6】 traps
    代码:https://github.com/JasenChao/xv6-labs.gitbacktrace题目要求实现backtrace来对堆栈上调用发生错误的地方进行跟踪。寄存器s0包含指向当前堆栈帧的指针,那么返回地址就位于帧指针的固定偏移量-8,前一个fp地址的偏移量为-16。在riscv.h文件中增加提示中的代码:staticinline......
  • 【XV6】 file system
    代码:https://github.com/JasenChao/xv6-labs.git支持大文件XV6目前只支持268个blocks大小的文件,一个block(BSIZE)为1024,文件块inode包含12个一级地址和1个二级地址,二级地址指向另一个block,其中存放了256个一级地址,因此一共是268个。题目要求支持大文件(65803个blocks),提示通过三级......
  • 【XV6】 locks
    代码:https://github.com/JasenChao/xv6-labs.git内存分配器单个空闲内存列表可能引起多个CPU的频繁锁争用,题目要求设计内存分配器,让每个CPU维护一个空闲内存列表,不同CPU的分配和释放可以并行执行,但如果一个CPU可用列表为空,而其他CPU可用列表不为空,则这个CPU必须窃取其他CPU的空......
  • 【XV6】 networking
    代码:https://github.com/JasenChao/xv6-labs.gitE1000网络设备驱动题目已经在kernel/e1000.c中给出了E1000的初始化函数和发送接收函数,要求完善发送和接收的功能。其他相关的代码,上层的网络协议在kernel/net.c和kernel/net.h中。PCI总线上搜索网卡的代码在kernel/pci.c中://t......
  • [MIT 6.S081] Lab: Copy-on-Write Fork for xv6
    Lab:Copy-on-WriteForkforxv6在这个实验中,我们要为xv6实现cowfork。Implementcopy-onwrite根据写时复制的方式,我们在复制页面的时候应该采用的是将父级的物理页面映射到子级的页面,因此我们需要修改kernel/vm.c中的uvmcopy,将页面复制修改为映射的方式,同时应当将......
  • MySQL表结构及数据对比工具:MySQL Utilities
    简介MySQLUtilities是一组基于python语言编写的python库的命令行实用工具集,依赖于python2.6。该工具提供了MySQL数据库运维工程中常用的一些工具,诸如克隆、复制、比较、差异、导出、导入、安装、配置、索引、磁盘查看等等。Linux下安装yuminstallmysql-utilities.noarch......