DragonOS/user/apps/shell/cmd.c

679 lines
15 KiB
C
Raw Normal View History

2022-05-24 08:37:28 +00:00
#include "cmd.h"
#include "cmd_help.h"
#include "cmd_test.h"
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>
#include <libsystem/syscall.h>
#include <signal.h>
#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <unistd.h>
2022-05-24 08:37:28 +00:00
#define MAX_PATH_LEN 4096
2022-05-24 08:37:28 +00:00
// 当前工作目录在main_loop中初始化
char *shell_current_path = NULL;
2022-05-24 08:37:28 +00:00
/**
* @brief shell
*
*/
struct built_in_cmd_t shell_cmds[] = {
新的内存管理模块 (#303) &emsp;&emsp;实现了具有优秀架构设计的新的内存管理模块,对内核空间和用户空间的内存映射、分配、释放、管理等操作进行了封装,使得内核开发者可以更加方便地进行内存管理。 &emsp;&emsp;内存管理模块主要由以下类型的组件组成: - **硬件抽象层(MemoryManagementArch)** - 提供对具体处理器架构的抽象,使得内存管理模块可以在不同的处理器架构上运行 - **页面映射器(PageMapper)**- 提供对虚拟地址和物理地址的映射,以及页表的创建、填写、销毁、权限管理等操作。分为两种类型:内核页表映射器(KernelMapper)和用户页表映射器(位于具体的用户地址空间结构中) - **页面刷新器(PageFlusher)** - 提供对页表的刷新操作(整表刷新、单页刷新、跨核心刷新) - **页帧分配器(FrameAllocator)** - 提供对页帧的分配、释放、管理等操作。具体来说,包括BumpAllocator、BuddyAllocator - **小对象分配器** - 提供对小内存对象的分配、释放、管理等操作。指的是内核里面的SlabAllocator (SlabAllocator的实现目前还没有完成) - **MMIO空间管理器** - 提供对MMIO地址空间的分配、管理操作。(目前这个模块待进一步重构) - **用户地址空间管理机制** - 提供对用户地址空间的管理。 - VMA机制 - 提供对用户地址空间的管理,包括VMA的创建、销毁、权限管理等操作 - 用户映射管理 - 与VMA机制共同作用,管理用户地址空间的映射 - **系统调用层** - 提供对用户空间的内存管理系统调用,包括mmap、munmap、mprotect、mremap等 - **C接口兼容层** - 提供对原有的C代码的接口,是的C代码能够正常运行。 除上面的新增内容以外,其它的更改内容: - 新增二进制加载器,以及elf的解析器 - 解决由于local_irq_save、local_irq_restore函数的汇编不规范导致影响栈行为的bug。 - 解决local_irq_save未关中断的错误。 - 修复sys_gettimeofday对timezone参数的处理的bug --------- Co-authored-by: kong <kongweichao@dragonos.org>
2023-07-22 08:27:02 +00:00
{"cd", shell_cmd_cd},
{"cat", shell_cmd_cat},
{"exec", shell_cmd_exec},
{"ls", shell_cmd_ls},
{"mkdir", shell_cmd_mkdir},
{"pwd", shell_cmd_pwd},
{"rm", shell_cmd_rm},
{"rmdir", shell_cmd_rmdir},
{"reboot", shell_cmd_reboot},
{"touch", shell_cmd_touch},
{"about", shell_cmd_about},
{"free", shell_cmd_free},
{"help", shell_help},
{"pipe", shell_pipe_test},
2023-09-12 16:58:01 +00:00
{"pipe2", shell_pipe2_test},
新的内存管理模块 (#303) &emsp;&emsp;实现了具有优秀架构设计的新的内存管理模块,对内核空间和用户空间的内存映射、分配、释放、管理等操作进行了封装,使得内核开发者可以更加方便地进行内存管理。 &emsp;&emsp;内存管理模块主要由以下类型的组件组成: - **硬件抽象层(MemoryManagementArch)** - 提供对具体处理器架构的抽象,使得内存管理模块可以在不同的处理器架构上运行 - **页面映射器(PageMapper)**- 提供对虚拟地址和物理地址的映射,以及页表的创建、填写、销毁、权限管理等操作。分为两种类型:内核页表映射器(KernelMapper)和用户页表映射器(位于具体的用户地址空间结构中) - **页面刷新器(PageFlusher)** - 提供对页表的刷新操作(整表刷新、单页刷新、跨核心刷新) - **页帧分配器(FrameAllocator)** - 提供对页帧的分配、释放、管理等操作。具体来说,包括BumpAllocator、BuddyAllocator - **小对象分配器** - 提供对小内存对象的分配、释放、管理等操作。指的是内核里面的SlabAllocator (SlabAllocator的实现目前还没有完成) - **MMIO空间管理器** - 提供对MMIO地址空间的分配、管理操作。(目前这个模块待进一步重构) - **用户地址空间管理机制** - 提供对用户地址空间的管理。 - VMA机制 - 提供对用户地址空间的管理,包括VMA的创建、销毁、权限管理等操作 - 用户映射管理 - 与VMA机制共同作用,管理用户地址空间的映射 - **系统调用层** - 提供对用户空间的内存管理系统调用,包括mmap、munmap、mprotect、mremap等 - **C接口兼容层** - 提供对原有的C代码的接口,是的C代码能够正常运行。 除上面的新增内容以外,其它的更改内容: - 新增二进制加载器,以及elf的解析器 - 解决由于local_irq_save、local_irq_restore函数的汇编不规范导致影响栈行为的bug。 - 解决local_irq_save未关中断的错误。 - 修复sys_gettimeofday对timezone参数的处理的bug --------- Co-authored-by: kong <kongweichao@dragonos.org>
2023-07-22 08:27:02 +00:00
{"kill", shell_cmd_kill},
2022-05-24 08:37:28 +00:00
};
// 总共的内建命令数量
const static int total_built_in_cmd_num = sizeof(shell_cmds) / sizeof(struct built_in_cmd_t);
2022-05-31 13:55:06 +00:00
/**
* @brief cwd与文件名进行拼接
*
* @param filename
* @param result_path_len
* @return char*
*/
static char *get_target_filepath(const char *filename, int *result_path_len)
{
char *file_path = NULL;
if (filename[0] != '/')
{
int cwd_len = strlen(shell_current_path);
2022-05-31 13:55:06 +00:00
// 计算文件完整路径的长度
*result_path_len = cwd_len + strlen(filename);
2022-05-31 13:55:06 +00:00
file_path = (char *)malloc(*result_path_len + 2);
2022-05-31 13:55:06 +00:00
memset(file_path, 0, *result_path_len + 2);
2022-05-31 13:55:06 +00:00
strncpy(file_path, shell_current_path, cwd_len);
2022-05-31 13:55:06 +00:00
// 在文件路径中加入斜杠
if (cwd_len > 1)
file_path[cwd_len] = '/';
2022-05-31 13:55:06 +00:00
// 拼接完整路径
strcat(file_path, filename);
}
else
{
*result_path_len = strlen(filename);
file_path = (char *)malloc(*result_path_len + 2);
memset(file_path, 0, *result_path_len + 2);
strncpy(file_path, filename, *result_path_len);
if (filename[(*result_path_len) - 1] != '/')
file_path[*result_path_len] = '/';
}
2022-05-31 13:55:06 +00:00
return file_path;
}
2022-05-24 08:37:28 +00:00
/**
* @brief
*
* @param main_cmd
* @return int
* -1
*/
int shell_find_cmd(char *main_cmd)
{
for (int i = 0; i < total_built_in_cmd_num; ++i)
{
if (strcmp(main_cmd, shell_cmds[i].name) == 0) // 找到对应的命令号
return i;
}
// 找不到该命令
return -1;
}
/**
* @brief shell内建的命令
*
* @param index
* @param argc
* @param argv
*/
void shell_run_built_in_command(int index, int argc, char **argv)
{
if (index >= total_built_in_cmd_num)
return;
// printf("run built-in command : %s\n", shell_cmds[index].name);
shell_cmds[index].func(argc, argv);
}
/**
* @brief cd命令:
*
* @param argc
* @param argv
* @return int
*/
2022-05-25 14:50:32 +00:00
int shell_cmd_cd(int argc, char **argv)
{
int current_dir_len = strlen(shell_current_path);
if (argc < 2)
{
shell_help_cd();
goto done;
}
// 进入当前文件夹
if (!strcmp(".", argv[1]))
goto done;
// 进入父目录
if (!strcmp("..", argv[1]))
{
// 当前已经是根目录
if (!strcmp("/", shell_current_path))
goto done;
// 返回到父目录
int index = current_dir_len - 1;
for (; index > 1; --index)
{
if (shell_current_path[index] == '/')
break;
}
shell_current_path[index] = '\0';
// printf("switch to \" %s \"\n", shell_current_path);
goto done;
}
int dest_len = strlen(argv[1]);
// 路径过长
if (dest_len >= SHELL_CWD_MAX_SIZE - 1)
{
printf("ERROR: Path too long!\n");
goto fail;
}
if (argv[1][0] == '/')
{
// ======进入绝对路径=====
int ec = chdir(argv[1]);
if (ec == -1)
ec = errno;
if (ec == 0)
{
// 获取新的路径字符串
char *new_path = (char *)malloc(MAX_PATH_LEN);
if (new_path==NULL) {
goto fail;
}
memset(new_path, 0, MAX_PATH_LEN);
getcwd(new_path, MAX_PATH_LEN);
2022-05-25 14:50:32 +00:00
// 释放原有的路径字符串的内存空间
free(shell_current_path);
shell_current_path = new_path;
return 0;
}
else
goto fail;
; // 出错则直接忽略
2022-05-25 14:50:32 +00:00
}
else // ======进入相对路径=====
2022-05-25 14:50:32 +00:00
{
int dest_offset = 0;
if (dest_len > 2)
{
if (argv[1][0] == '.' && argv[1][1] == '/') // 相对路径
dest_offset = 2;
}
2022-05-29 06:36:46 +00:00
int new_len = current_dir_len + dest_len - dest_offset;
2022-05-25 14:50:32 +00:00
if (new_len >= SHELL_CWD_MAX_SIZE - 1)
{
printf("ERROR: Path too long!\n");
goto fail;
}
// 拼接出新的字符串
char *new_path = (char *)malloc(new_len + 2);
memset(new_path, 0, new_len);
2022-05-25 14:50:32 +00:00
strncpy(new_path, shell_current_path, current_dir_len);
if (current_dir_len > 1)
new_path[current_dir_len] = '/';
strcat(new_path, argv[1] + dest_offset);
2022-09-07 15:25:39 +00:00
int x = chdir(new_path);
if (x == 0) // 成功切换目录
2022-05-25 14:50:32 +00:00
{
free(new_path);
2022-05-25 14:50:32 +00:00
free(shell_current_path);
char * pwd = malloc(MAX_PATH_LEN);
if (pwd==NULL) {
goto fail;
}
memset(pwd, 0, MAX_PATH_LEN);
getcwd(pwd, MAX_PATH_LEN);
shell_current_path = pwd;
2022-05-25 14:50:32 +00:00
goto done;
}
else
{
free(new_path);
2022-05-25 14:50:32 +00:00
printf("ERROR: Cannot switch to directory: %s\n", new_path);
goto fail;
}
}
fail:;
done:;
// 释放参数所占的内存
free(argv);
return 0;
}
2022-05-24 08:37:28 +00:00
/**
* @brief
*
* @param argc
* @param argv
* @return int
*/
2022-05-25 14:50:32 +00:00
int shell_cmd_ls(int argc, char **argv)
{
2022-05-29 06:36:46 +00:00
struct DIR *dir = opendir(shell_current_path);
2022-05-30 09:39:45 +00:00
2022-05-29 06:36:46 +00:00
if (dir == NULL)
return -1;
struct dirent *buf = NULL;
// printf("dir=%#018lx\n", dir);
while (1)
{
buf = readdir(dir);
2022-05-30 09:39:45 +00:00
if (buf == NULL)
2022-05-29 06:36:46 +00:00
break;
2022-05-30 09:39:45 +00:00
2022-05-29 06:36:46 +00:00
int color = COLOR_WHITE;
if (buf->d_type == DT_DIR)
2022-05-29 06:36:46 +00:00
color = COLOR_YELLOW;
else if (buf->d_type == DT_REG)
2022-05-29 06:36:46 +00:00
color = COLOR_INDIGO;
else if (buf->d_type == DT_BLK || buf->d_type == DT_CHR)
2022-09-12 15:56:31 +00:00
color = COLOR_GREEN;
2022-05-30 09:39:45 +00:00
2022-05-29 06:36:46 +00:00
char output_buf[256] = {0};
sprintf(output_buf, "%s ", buf->d_name);
put_string(output_buf, color, COLOR_BLACK);
}
printf("\n");
closedir(dir);
2022-05-30 09:39:45 +00:00
2022-05-31 13:55:06 +00:00
if (argv != NULL)
2022-05-30 09:39:45 +00:00
free(argv);
2022-05-25 14:50:32 +00:00
return 0;
}
2022-05-24 08:37:28 +00:00
/**
* @brief
*
* @param argc
* @param argv
* @return int
*/
int shell_cmd_pwd(int argc, char **argv)
{
if (shell_current_path)
printf("%s\n", shell_current_path);
2022-05-31 13:55:06 +00:00
if (argv != NULL)
2022-05-30 09:39:45 +00:00
free(argv);
return 0;
2022-05-24 08:37:28 +00:00
}
/**
* @brief
*
* @param argc
* @param argv
* @return int
*/
2022-05-30 09:39:45 +00:00
int shell_cmd_cat(int argc, char **argv)
{
2022-05-31 13:55:06 +00:00
int path_len = 0;
char *file_path = get_target_filepath(argv[1], &path_len);
2022-05-30 09:39:45 +00:00
// 打开文件
int fd = open(file_path, O_RDONLY);
if (fd <= 0)
{
printf("ERROR: Cannot open file: %s, fd=%d\n", file_path, fd);
return -1;
}
2022-05-30 09:39:45 +00:00
// 获取文件总大小
int file_size = lseek(fd, 0, SEEK_END);
// 将文件指针切换回文件起始位置
lseek(fd, 0, SEEK_SET);
char *buf = (char *)malloc(512);
2022-05-30 09:39:45 +00:00
while (file_size > 0)
{
memset(buf, 0, 512);
2022-05-30 09:39:45 +00:00
int l = read(fd, buf, 511);
if (l < 0)
{
printf("ERROR: Cannot read file: %s, errno = %d\n", file_path, errno);
return -1;
}
if (l == 0)
break;
2022-05-30 09:39:45 +00:00
buf[l] = '\0';
file_size -= l;
printf("%s", buf);
}
close(fd);
free(buf);
free(file_path);
2022-05-31 13:55:06 +00:00
if (argv != NULL)
free(argv);
return 0;
2022-05-30 09:39:45 +00:00
}
2022-05-24 08:37:28 +00:00
/**
* @brief
*
* @param argc
* @param argv
* @return int
*/
2022-07-06 16:55:33 +00:00
int shell_cmd_touch(int argc, char **argv)
{
int path_len = 0;
2022-07-07 02:27:40 +00:00
char *file_path;
bool alloc_full_path = false;
2022-07-07 02:27:40 +00:00
if (argv[1][0] == '/')
file_path = argv[1];
else
{
file_path = get_target_filepath(argv[1], &path_len);
alloc_full_path = true;
}
2022-07-06 16:55:33 +00:00
// 打开文件
int fd = open(file_path, O_CREAT);
2022-07-07 02:27:40 +00:00
switch (fd)
{
case -ENOENT:
put_string("Parent dir not exists.\n", COLOR_RED, COLOR_BLACK);
break;
default:
break;
}
2022-07-06 16:55:33 +00:00
close(fd);
if (argv != NULL)
free(argv);
if (alloc_full_path)
free(file_path);
return 0;
2022-07-06 16:55:33 +00:00
}
2022-05-24 08:37:28 +00:00
/**
* @brief
*
* @param argc
* @param argv
* @return int
*/
int shell_cmd_mkdir(int argc, char **argv)
{
int result_path_len = -1;
char *full_path = NULL;
bool alloc_full_path = false;
if (argv[1][0] == '/')
full_path = argv[1];
else
{
full_path = get_target_filepath(argv[1], &result_path_len);
alloc_full_path = true;
}
2022-09-13 08:17:35 +00:00
// printf("mkdir: full_path = %s\n", full_path);
int retval = mkdir(full_path, 0);
2022-07-06 16:55:33 +00:00
if (argv != NULL)
free(argv);
if (alloc_full_path)
free(full_path);
return retval;
}
2022-05-24 08:37:28 +00:00
/**
* @brief
*
* @param argc
* @param argv
* @return int
*/
2022-09-12 15:56:31 +00:00
int shell_cmd_rmdir(int argc, char **argv)
{
char *full_path = NULL;
2022-09-12 15:56:31 +00:00
int result_path_len = -1;
bool alloc_full_path = false;
2022-09-12 15:56:31 +00:00
if (argv[1][0] == '/')
full_path = argv[1];
else
{
2022-09-12 15:56:31 +00:00
full_path = get_target_filepath(argv[1], &result_path_len);
alloc_full_path = true;
}
2022-09-12 15:56:31 +00:00
int retval = rmdir(full_path);
if (retval != 0)
printf("Failed to remove %s, retval=%d\n", full_path, retval);
2022-09-13 08:17:35 +00:00
// printf("rmdir: path=%s, retval=%d\n", full_path, retval);
2022-09-12 15:56:31 +00:00
if (argv != NULL)
free(argv);
if (alloc_full_path)
free(full_path);
return retval;
}
2022-09-12 15:56:31 +00:00
/**
* @brief
*
* @param argc
* @param argv
* @return int
*/
int shell_cmd_rm(int argc, char **argv)
{
char *full_path = NULL;
int result_path_len = -1;
int retval = 0;
bool alloc_full_path = false;
if (argv[1][0] == '/')
full_path = argv[1];
else
{
full_path = get_target_filepath(argv[1], &result_path_len);
alloc_full_path = true;
}
retval = rm(full_path);
// printf("rmdir: path=%s, retval=%d\n", full_path, retval);
if (retval != 0)
printf("Failed to remove %s, retval=%d\n", full_path, retval);
if (alloc_full_path)
free(full_path);
if (argv != NULL)
free(argv);
2022-09-12 15:56:31 +00:00
return retval;
}
2022-05-24 08:37:28 +00:00
/**
* @brief
*
* @param argc
* @param argv
* @return int
*/
2022-05-31 13:55:06 +00:00
int shell_cmd_exec(int argc, char **argv)
{
pid_t pid = fork();
int retval = 0;
2022-06-07 15:18:26 +00:00
// printf(" pid=%d \n",pid);
2022-05-24 08:37:28 +00:00
2022-05-31 13:55:06 +00:00
if (pid == 0)
{
2022-05-31 13:55:06 +00:00
// 子进程
int path_len = 0;
char *file_path = get_target_filepath(argv[1], &path_len);
// printf("before execv, path=%s, argc=%d\n", file_path, argc);
char **real_argv = NULL;
if (argc > 1)
{
real_argv = &argv[1];
}
execv(file_path, real_argv);
// printf("after execv, path=%s, argc=%d\n", file_path, argc);
2022-05-31 13:55:06 +00:00
free(argv);
free(file_path);
exit(-1);
2022-05-31 13:55:06 +00:00
}
else
{
// 如果不指定后台运行,则等待退出
if (strcmp(argv[argc - 1], "&") != 0)
waitpid(pid, &retval, 0);
else
printf("[1] %d\n", pid); // 输出子进程的pid
2022-05-31 13:55:06 +00:00
free(argv);
}
}
int shell_cmd_about(int argc, char **argv)
{
if (argv != NULL)
free(argv);
int aac = 0;
char **aav;
unsigned char input_buffer[INPUT_BUFFER_SIZE] = {0};
strcpy(input_buffer, "exec /bin/about.elf\0");
parse_command(input_buffer, &aac, &aav);
2022-05-31 13:55:06 +00:00
return shell_cmd_exec(aac, aav);
}
int shell_cmd_kill(int argc, char **argv)
{
int retval = 0;
if (argc < 2)
{
printf("Usage: Kill <pid>\n");
retval = -EINVAL;
goto out;
}
retval = kill(atoi(argv[1]), SIGKILL);
out:;
free(argv);
return retval;
2022-05-31 13:55:06 +00:00
}
2022-05-24 08:37:28 +00:00
/**
* @brief
*
* @param argc
* @param argv
* @return int
*/
2022-05-25 06:38:15 +00:00
int shell_cmd_reboot(int argc, char **argv)
{
return syscall_invoke(SYS_REBOOT, 0, 0, 0, 0, 0, 0);
}
2022-08-07 13:17:02 +00:00
int shell_cmd_free(int argc, char **argv)
{
int retval = 0;
if (argc == 2 && strcmp("-m", argv[1]) != 0)
{
retval = -EINVAL;
printf("Invalid argument: %s\n", argv[1]);
goto done;
}
struct mstat_t mst = {0};
retval = mstat(&mst);
2022-09-12 15:56:31 +00:00
if (retval != 0)
2022-08-07 13:17:02 +00:00
{
printf("Failed: retval=%d", retval);
goto done;
}
printf("\ttotal\tused\tfree\tshared\tcache\tavailable\n");
printf("Mem:\t");
2022-09-12 15:56:31 +00:00
if (argc == 1) // 按照kb显示
2022-08-07 13:17:02 +00:00
{
printf("%ld\t%ld\t%ld\t%ld\t%ld\t%ld\t\n", mst.total, mst.used, mst.free, mst.shared,
mst.cache_used, mst.available);
2022-08-07 13:17:02 +00:00
}
2022-09-12 15:56:31 +00:00
else // 按照MB显示
2022-08-07 13:17:02 +00:00
{
printf("%ld\t%ld\t%ld\t%ld\t%ld\t%ld\t\n", mst.total >> 10, mst.used >> 10, mst.free >> 10, mst.shared >> 10,
mst.cache_used >> 10, mst.available >> 10);
2022-08-07 13:17:02 +00:00
}
2022-09-12 15:56:31 +00:00
2022-08-07 13:17:02 +00:00
done:;
if (argv != NULL)
free(argv);
return retval;
}
/**
* @brief shell命令
*
* @param buf
* @param argc
* @param argv
* @return int
*/
int parse_command(char *buf, int *argc, char ***argv)
{
// printf("parse command\n");
int index = 0; // 当前访问的是buf的第几位
// 去除命令前导的空格
while (index < INPUT_BUFFER_SIZE && buf[index] == ' ')
++index;
// 如果去除前导空格后第一项为0x00则归为空命令
新的内存管理模块 (#303) &emsp;&emsp;实现了具有优秀架构设计的新的内存管理模块,对内核空间和用户空间的内存映射、分配、释放、管理等操作进行了封装,使得内核开发者可以更加方便地进行内存管理。 &emsp;&emsp;内存管理模块主要由以下类型的组件组成: - **硬件抽象层(MemoryManagementArch)** - 提供对具体处理器架构的抽象,使得内存管理模块可以在不同的处理器架构上运行 - **页面映射器(PageMapper)**- 提供对虚拟地址和物理地址的映射,以及页表的创建、填写、销毁、权限管理等操作。分为两种类型:内核页表映射器(KernelMapper)和用户页表映射器(位于具体的用户地址空间结构中) - **页面刷新器(PageFlusher)** - 提供对页表的刷新操作(整表刷新、单页刷新、跨核心刷新) - **页帧分配器(FrameAllocator)** - 提供对页帧的分配、释放、管理等操作。具体来说,包括BumpAllocator、BuddyAllocator - **小对象分配器** - 提供对小内存对象的分配、释放、管理等操作。指的是内核里面的SlabAllocator (SlabAllocator的实现目前还没有完成) - **MMIO空间管理器** - 提供对MMIO地址空间的分配、管理操作。(目前这个模块待进一步重构) - **用户地址空间管理机制** - 提供对用户地址空间的管理。 - VMA机制 - 提供对用户地址空间的管理,包括VMA的创建、销毁、权限管理等操作 - 用户映射管理 - 与VMA机制共同作用,管理用户地址空间的映射 - **系统调用层** - 提供对用户空间的内存管理系统调用,包括mmap、munmap、mprotect、mremap等 - **C接口兼容层** - 提供对原有的C代码的接口,是的C代码能够正常运行。 除上面的新增内容以外,其它的更改内容: - 新增二进制加载器,以及elf的解析器 - 解决由于local_irq_save、local_irq_restore函数的汇编不规范导致影响栈行为的bug。 - 解决local_irq_save未关中断的错误。 - 修复sys_gettimeofday对timezone参数的处理的bug --------- Co-authored-by: kong <kongweichao@dragonos.org>
2023-07-22 08:27:02 +00:00
if (!buf[index])
return -1;
// 计算参数数量
for (int i = index; i < (INPUT_BUFFER_SIZE - 1); ++i)
{
// 到达了字符串末尾
if (!buf[i])
break;
if (buf[i] != ' ' && (buf[i + 1] == ' ' || buf[i + 1] == '\0'))
++(*argc);
}
// printf("\nargc=%d\n", *argc);
// 为指向每个指令的指针分配空间
新的内存管理模块 (#303) &emsp;&emsp;实现了具有优秀架构设计的新的内存管理模块,对内核空间和用户空间的内存映射、分配、释放、管理等操作进行了封装,使得内核开发者可以更加方便地进行内存管理。 &emsp;&emsp;内存管理模块主要由以下类型的组件组成: - **硬件抽象层(MemoryManagementArch)** - 提供对具体处理器架构的抽象,使得内存管理模块可以在不同的处理器架构上运行 - **页面映射器(PageMapper)**- 提供对虚拟地址和物理地址的映射,以及页表的创建、填写、销毁、权限管理等操作。分为两种类型:内核页表映射器(KernelMapper)和用户页表映射器(位于具体的用户地址空间结构中) - **页面刷新器(PageFlusher)** - 提供对页表的刷新操作(整表刷新、单页刷新、跨核心刷新) - **页帧分配器(FrameAllocator)** - 提供对页帧的分配、释放、管理等操作。具体来说,包括BumpAllocator、BuddyAllocator - **小对象分配器** - 提供对小内存对象的分配、释放、管理等操作。指的是内核里面的SlabAllocator (SlabAllocator的实现目前还没有完成) - **MMIO空间管理器** - 提供对MMIO地址空间的分配、管理操作。(目前这个模块待进一步重构) - **用户地址空间管理机制** - 提供对用户地址空间的管理。 - VMA机制 - 提供对用户地址空间的管理,包括VMA的创建、销毁、权限管理等操作 - 用户映射管理 - 与VMA机制共同作用,管理用户地址空间的映射 - **系统调用层** - 提供对用户空间的内存管理系统调用,包括mmap、munmap、mprotect、mremap等 - **C接口兼容层** - 提供对原有的C代码的接口,是的C代码能够正常运行。 除上面的新增内容以外,其它的更改内容: - 新增二进制加载器,以及elf的解析器 - 解决由于local_irq_save、local_irq_restore函数的汇编不规范导致影响栈行为的bug。 - 解决local_irq_save未关中断的错误。 - 修复sys_gettimeofday对timezone参数的处理的bug --------- Co-authored-by: kong <kongweichao@dragonos.org>
2023-07-22 08:27:02 +00:00
*argv = (char **)malloc(sizeof(char **) * (*argc + 1));
memset(*argv, 0, sizeof(char **) * (*argc + 1));
// 将每个命令都单独提取出来
for (int i = 0; i < *argc && index < INPUT_BUFFER_SIZE; ++i)
{
// 提取出命令,以空格作为分割
*((*argv) + i) = &buf[index];
while (index < (INPUT_BUFFER_SIZE - 1) && buf[index] && buf[index] != ' ')
++index;
buf[index++] = '\0';
// 删除命令间多余的空格
while (index < INPUT_BUFFER_SIZE && buf[index] == ' ')
++index;
// printf("%s\n", (*argv)[i]);
}
// 以第一个命令作为主命令,查找其在命令表中的编号
return shell_find_cmd(**argv);
2022-05-25 06:38:15 +00:00
}