Linux相关概念和易错知识点(16)(Shell原理、进程属性和环境变量表的联系)

Shell原理及其模拟实现

认识进程exec系列函数、命令行参数列表、环境变量之后,我们可以尝试理解一下Shell的原理,将各方知识串联起来,让Shell跑起来才能真正理解这些概念。我会以模拟Shell执行的原理模拟一个Shell。途中配上相关讲解。

1.Shell进程的创建

Shell是用户启动后系统为我们启动的第一个进程,用于启动CLI程序,为后面我们的命令行操作做准备,在Linux中,这个Shell具体程序是bash。

bash在/usr/bin/bash下,每一个用户登陆后都会先在内存中创建PCB,再从硬盘中将代码和数据读到内存中来,就和其他进程一样。

2.环境变量的配置

我们从上面那张图中就能发现,bash并不是在每个用户的home目录下,而是在公共区域。同一个程序却能生成针对不同用户登陆的bash,这和环境变量的配置有直接关系。当创建bash读取硬盘数据时,bash会从登陆用户的home目录下读取.bash_profile和.bashrc,里面有环境变量配置的相关信息。但是注意,配置文件里面并不包含所有的环境变量,它是局部的,还有一些环境变量是从bash的父进程继承下来的或是后面生成的。

下面是配置文件里面的一些内容,这些环境变量都会被bash导入到内存中形成一份临时环境变量表,在代码或命令行中修改环境变量就是修改的内存中的环境变量,只要不改硬盘文件,就不会修改默认环境变量。

环境变量是进程中最基础的部分,所有进程启动时都要导入一份环境变量

对于Shell进程,方式有:父进程继承、读取配置文件、后续自动生成(HOME -> cwd -> PWD)

对于其它父子进程而言方式有:父进程继承

当然我们都可以手动添加环境变量。

我们要模拟Shell进程,这意味着我们要完成父进程继承、读取配置文件、后续自动生成。这显然超出了我们的能力,把上面的实现了基本上我们就写出了一个系统。但我们可以模拟普通进程的导入环境变量,拷贝一份环境变量表做模拟。

这里有个易错点,一定要注意。main函数的三个参数(int argc, char** argv, char** env)的env是一个局部变量,且只有我们写int main(int argc, char** argv, char** env)时env才会被导入到函数栈帧中(写了参数调用的main函数就不一样了),后续如果要用env这个变量就要自己手动传参。env里面就存着环境变量。但程序中还有一个变量char** environ,它是全局的,且我们只要extern之后就可以在任何地方调用。

我们最终采用environ方案。

对于这个模拟的Shell程序来讲,我们手动显式实现了一个环境变量表,而我们需要搞清楚这个程序本身还有一个环境变量,那才是该程序真正的环境变量表,我们environ本质就是从该程序的本身的变量表拷贝过来的。

3.对环境变量表、命令行参数列表、进程属性之间的关系解读

在我们的模拟代码中,环境变量表存在于Shell程序的全局区域,可以被任意调用,在进程角度看来环境变量表属于代码和数据部分,本质上是存在于mm_struct管理的映射区域的,这和进程属性直接存在PCB中有本质区别。命令行参数argc和argv都是属于代码和数据的。

(1)对mm_struct进一步解释

进程的mm_struct里面管理进程地址空间及其映射的一部分物理空间,这部分空间里面存的是进程的代码和数据,而除了代码和数据是不会存在于进程地址空间里的(比如进程属性),因此我们在栈区、堆区等地方是找不到进程的属性struct mm_struct* mm或者是cwd的,因为它们是PCB的属性,直接存在物理内存中,没有页表映射。操作系统直接管理,用户永远无法也无需直接获取它们的地址。

综上,进程属性本质上是属于PCB,由Kernel直接管理的,我们永远拿不到它们的物理地址。而相比之下,环境变量属于进程的代码和数据,存在于mm_struct管理的物理空间,对应进程地址空间的位置是高于栈区的。

(2)环境变量PWD和进程属性cwd的关系

在这里我们需要了解一些后续生成的环境变量是如何来的。其中最重要的就是PWD和cwd的关系!

当读取配置文件后,HOME被配置好了,之后cwd会使用chdir修改自己的当前工作目录,再之后会借助cwd里面的数据使用putenv这个函数创建一个新的环境变量PWD。PWD环境变量是借助cwd这个进程属性来初始化的。

注意,只有Shell进程创建时会干这事。其余进程创建时都会使用写时拷贝的技术继承父进程的环境变量表,不会有任何读取文件或是自动添加环境变量操作。

3.打印命令行提示符

命令行提示符的格式是:[用户名(USER)@主机名(HOSTNAME) 当前路径(PWD)]$/#

这部分主要就是字符串处理相关的知识。获取的环境变量是从我们自己拷贝得到的环境变量表中取,而不是在该进程原本的环境变量表中取。

注意C/C++混编的情况下string有着避免野指针的优势,我们可以在函数里定义string,然后返回它,这样做不会出现和数组那样的野指针问题,会自动初始化一个新的string。当然我们要注意C/C++之间的转换,如c_str()这种接口要熟悉

4.获取并分析用户命令

这是一个非常容易犯错误的地方。全局的argv和argc、获取用户命令的CommandBuffer每次都要清空数据,之后用fgets安全读入一行,strtok分割字符串并存入argv中。

注意回车符也会被读入,读入后的下一个字符被标记为'\0',读取结束,因此我们要手动处理字符串中有回车的情况,将它改成'\0'。

5.执行命令

(1)子进程执行命令

借助execvp和argv,我们可以实现子进程执行程序,父进程wait子进程。这样做的好处是子进程执行失败完全不会影响父进程的安全性,如果直接让父进程执行所有命令,那么如果出了一个较严重的错误,父进程就直接被挂掉了,这显然不是我们希望看到的。

(2)内建命令

内建命令用于修改当前进程环境变量的值或者要访问只有该进程能访问的数据。如echo能访问本地变量,cd要修改本进程的环境变量。因为子进程执行指令没办法访问父进程的数据,进程之间的独立性决定了这类命令只能直接由父进程执行。

思路就是穷举法,将需要的命令手动在父进程处理。注意chdir修改的是进程属性cwd,进程属性和环境变量之间在进程运行时是各改个的,只有在Shell创建时的初始化时才存在关联。它们之间的同步需要手动维护。

我们从一个更加底层的角度上来想,进程属性属于PCB、系统直接管理对象,而环境变量是程序代码和数据的一部分。当修改一边时,另一边理所应当保持独立,只不过Shell内部维护导致我们大部分看上去是同步的,但事实上修改cwd后PWD依然不变,cwd和PWD都只是数据而已,并没什么大不了的,因此我们需要手动putenv。 

在有的时候会发现存在不同步的情况。比如在Shell进程内切换用户,环境变量表会变,因为切换用户会重新读取配置文件,但进程始终是同一个,进程创建者不变。

注意getenv和putenv我们都要自己实现,因为系统给我们的这些函数都只会到进程自带的环境变量表中查找,我们要实现到自己的环境变量中查找、添加等,都要自己写。

最后可以使用全局的lastcode存储退出码,同样使用内建命令处理echo来获得退出码,底层逻辑是一样的,也很简单,这里就不再讲述了。唯一需要注意的是当没有找到命令时,错误信息和退出码是要在exec函数后面更新的,exec函数执行成功就不会执行下面的语句,执行失败就要。

总结:

我们通过Shell的原理和模拟实现主要是为了搞清进程属性和环境变量表、命令行参数表的关系。我们发现,一个进程 = PCB + 代码和数据,PCB可以管理这些代码和数据。其中进程属性直接是PCB的成员,被系统直接管理。而环境变量表、退出码、命令行参数列表本质都是存在代码和数据中的,也可以叫程序的上下文中。正是这样的差异导致了环境变量和进程属性的独立性,也帮助我们理解环境变量表是如何传给子进程的,命令行参数从读取到传给argv的过程是怎样的,以及退出码是如何读取到的。整个Shell的知识点都被串联起来了,很值得我们消化。

全部代码:


#include <sys/wait.h>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <unistd.h>
using namespace std;

const int basenum = 100;
const int basesize = 1024;

//命令行参数列表和环境变量表放在Shell的代码和数据中
char* genv[basenum];
char* gargv[basesize];
int gargc;

//存储即将更新的PWD,维护环境变量的更新,保持cwd和PWD的同步
int posPWD = 0;//随时存储PWD的位置,方便后续找到
char newPWD[basesize];



int lastcode;//存储退出码

void InitEnv()//初始化环境变量表,一般子进程是这样继承的,Shell进程是从文件读配置文件的,这里简略一点
{
    extern char** environ;
    for(int curi = 0; environ[curi]; curi++)
    {
        genv[curi] = (char*)malloc(strlen(environ[curi]) + 1);//不用sizeof,它会按指针大小算
        memcpy(genv[curi], environ[curi], strlen(environ[curi]) + 1);
    }
}

string GetEnv(string oneEnv)//用string防止不必要全局字符串,值拷贝避免野指针
{
    for(size_t i = 0; i < basenum; i++)
    {
        for(size_t j = 0; j < oneEnv.size(); j++)
        {
            if(oneEnv[j] != genv[i][j])//有不一样的就说明不匹配
                break;
            if(j == oneEnv.size() - 1)//全部相等
            {
                string User;
                for(size_t k = oneEnv.size() + 1; genv[i][k]; k++)
                    User += genv[i][k];
                posPWD = i;//存储PWD信息,方便后续找到
                return User;
            }
        }
    }
    
    return nullptr;
}


void PrintCommandLine()
{
    string GetUser = GetEnv("USER");
    string GetPwd = GetEnv("PWD");
    string GetHostName = GetEnv("HOSTNAME");

    if((GetPwd = string(GetPwd.begin() + GetPwd.rfind('/'), GetPwd.end())).size() > 1)//表达式的返回值是string,如果返回值不是根目录,都要把/删掉
        GetPwd.erase(0, 1);

    printf("[%s@%s %s]%c ", GetUser.c_str(), GetHostName.c_str(), GetPwd.c_str(), GetUser == "root" ? '#' : '$');
}


void GetCommandLine(char* CommandBuffer)//使用值拷贝string防止出现野指针
{
    memset(CommandBuffer, '\0', basesize);//每次都初始化读取用的字符串
    fgets(CommandBuffer, basesize, stdin);
    CommandBuffer[strlen(CommandBuffer) - 1] = '\0';
}

void ParseCommandLine(char* CommandBuffer)
{
    gargc = 0;//每次解析命令前先把上一个命令的信息删除
    memset(gargv, '\0', basesize);
    if(strlen(CommandBuffer) == 0)
        return;
    gargv[gargc++] = strtok(CommandBuffer, " ");
    while((bool)(gargv[gargc++] = strtok(nullptr, " ")));
    gargc--;
}

void ExecuteCommand()//没有指令的情况走不到这个函数
{
    pid_t ret = fork();
    if(ret == 0)
    {
        execvpe(gargv[0], gargv, genv);
        printf("-bash: %s: command not found\n", gargv[0]);
        lastcode = 1;
    }
    else
    {
        int status = 0;
        waitpid(ret, &status, 0);
        if(!WIFEXITED(status))
            lastcode =  1;
        else
            lastcode = WEXITSTATUS(status);
    }
}

bool CheckAndExecBuiltCommand()
{
    if(gargc == 0)//如果没有指令,直接进行下一轮循环,通过return true省的走下一个函数
        return true;
    

    if(strcmp("cd", gargv[0]) == 0)
    {
        if(gargc == 2)
        {
            memset(newPWD, '\0', basesize);
            chdir(gargv[1]);
            snprintf(newPWD, basesize, "PWD=%s", gargv[1]);
            genv[(GetEnv("PWD"), posPWD)] = newPWD;//每次调用GetEnv都会刷新posPWD的位置,用逗号表达式的特性实现
            lastcode = 0;
            return true;
        }
        return false;
    }

    if(strcmp("export", gargv[0]) == 0)
    {
        if(gargc == 2)
        {
            int curi = 0;
            while(genv[curi]) curi++;
            genv[curi] = (char*)malloc(strlen(gargv[1]) + 1);
            memcpy(genv[curi], gargv[1], strlen(gargv[1]) + 1);
            lastcode = 0;
            return true;
        }
        return false;
    }

    if(strcmp("env", gargv[0]) == 0)
    {
        if(gargc == 1)
        {
            for(int i = 0; genv[i]; i++)
                printf("%s\n", genv[i]);
            lastcode = 0;
            return true;
        }
        return false;
    }

    if(strcmp("echo", gargv[0]) == 0)
    {
        if(gargc == 2 && strcmp("$?", gargv[1]) == 0)
            printf("%d\n", lastcode);
        lastcode = 0;
        return true;
    }

    return false;
}



int main()
{
    char CommandBuffer[basesize] = { 0 };

    InitEnv();
    while(true)
    {
        PrintCommandLine();//打印命令行提示符
        GetCommandLine(CommandBuffer);//读取命令
        ParseCommandLine(CommandBuffer);//解析命令至argc和argv中
        if(CheckAndExecBuiltCommand()) continue;
        ExecuteCommand();
    }

    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/901092.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

InnoDB 存储引擎<一>InnoDB简介与MySQL存储架构及相关数据结构

目录 回顾MySQL架构 InnoDB简介 ​MySQL存储结构 回顾MySQL架构 对MySQL架构图的总结: MySQL服务器是以网络服务的方式对外提供数据库服务的&#xff0c;我们使用的应用程序以及客户端统称为外部程序。 外部程序通过发送网络请求的方式来连接MySQL服务器&#xff0c;这时首先每…

Leetcode239. 滑动窗口最大值

问题描述&#xff1a; 给你一个整数数组 nums&#xff0c;有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的 k 个数字。滑动窗口每次只向右移动一位。 返回 滑动窗口中的最大值 。 示例 1&#xff1a; 输入&#xff1a;nums [1,3,…

Python爬虫教程:从入门到精通

Python爬虫教程&#xff1a;从入门到精通 前言 在信息爆炸的时代&#xff0c;数据是最宝贵的资源之一。Python作为一种简洁而强大的编程语言&#xff0c;因其丰富的库和框架&#xff0c;成为了数据爬取的首选工具。本文将带您深入了解Python爬虫的基本概念、实用技巧以及应用…

文件下载漏洞

文件安全 文件下载 常见敏感信息路径 Windows C:\boot.ini //查看系统版本 C:\Windows\System32\inetsrv\MetaBase.xml //IIS配置文件 C:\Windows\repair\sam //存储系统初次安装的密码 C:\Program Files\mysql\my.ini //Mysql配置 C:\Program Files\mysql\data\mysql\user.…

Python小游戏13——植物大战僵尸

代码 import random import time # 植物类 class Plant: def __init__(self, name, health): self.name name self.health health def is_alive(self): return self.health > 0 # 僵尸类 class Zombie: def __init__(self, name, health): self.name name self.health h…

Tornado简单使用

Tornado简单使用 1 介绍 Tornado 是一个基于Python的Web服务框架和 异步网络库&#xff0c;它最初由 FriendFeed 开发&#xff0c;后来被 Facebook 收购并开源&#xff0c;通过利用非阻塞网络 I/O, Tornado 可以承载成千上万的活动连接&#xff0c;完美的实现了 长连接、WebS…

关于 Linux 内核“合规要求”与俄罗斯制裁的一些澄清

原文&#xff1a;Michael Larabel - 2024.10.24 当 一些俄罗斯的 Linux 开发者被从内核的 MAINTAINERS 文件中移除 时&#xff0c;原因被描述为“合规要求”&#xff0c;但并未明确这些要求具体涉及什么内容。随后&#xff0c;Linus Torvalds 对此发表了评论&#xff0c;明确指…

SIP 业务举例之 三方通话:邀请第三方加入的信令流程

目录 1. 3-Way Conference - Third Party Is Added 简介 2. RFC5359 的 3-Way Conference - Third Party Is Added 信令流程 3. 3-Way Conference - Third Party Is Added 总结 博主wx:yuanlai45_csdn 博主qq:2777137742 想要 深入学习 5GC IMS 等通信知识(加入 51学通信)…

[bug] vllm 0.6.1 RuntimeError: operator torchvision::nms does not exist

[bug] vllm 0.6.1 RuntimeError: operator torchvision::nms does not exist 环境 python 3.10 torch 2.4.0cu118 torchvision 0.19.0cu118 vllm 0.6.1.post2cu118问题详情 if torch._C._d…

Spring Boot框架中小企业设备监控系统开发

3系统分析 3.1可行性分析 通过对本中小企业设备管理系统实行的目的初步调查和分析&#xff0c;提出可行性方案并对其一一进行论证。我们在这里主要从技术可行性、经济可行性、操作可行性等方面进行分析。 3.1.1技术可行性 本中小企业设备管理系统采用Spring Boot框架&#xff0…

ubuntu20.04上使用 Verdaccio 搭建 npm 私有仓库

安装nvm 首先安装必要的工具&#xff1a; apt update apt install curl下载并执行nvm安装脚本&#xff1a; curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.39.7/install.sh | bash添加环境变量&#xff08;如果安装脚本没有自动添加&#xff09;。编辑 ~/.bash…

java项目之基于web的智慧社区设计与实现(springboot)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于springboot的基于web的智慧社区设计与实现。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 基于web的智…

使用TimeShift备份和恢复Ubuntu Linux

您是否曾经想过如何备份和恢复您的Ubuntu或Debian系统&#xff1f;TimeShift是一个强大的备份和还原工具。TimeShift允许您创建系统快照&#xff0c;提供了一种在出现意外问题或系统故障时恢复到先前状态的简便方式。您可以使用RSYNC或BTRFS创建快照。 有了这个介绍&#xff0…

萤石设备视频接入平台EasyCVR私有化视频平台变电站如何实现远程集中监控?

一、方案背景 随着城市经济的发展和电力系统的改造&#xff0c;变电站的数量和规模逐渐增加&#xff0c;对变电站的安全管理和监控需求也越来越高。视频监控系统作为重要的安全管理手段&#xff0c;在变电站中起到了关键的作用。 目前青犀视频研发的萤石设备视频接入平台EasyC…

【NOIP普及组】 选数

【NOIP普及组】 选数 &#x1f490;The Begin&#x1f490;点点关注&#xff0c;收藏不迷路&#x1f490; 已知 n 个整数 x1,x2,…,xn&#xff0c;以及一个整数 k&#xff08;k&#xff1c;n&#xff09;。从 n 个整数中任选 k 个整数相加&#xff0c;可分别得到一系列的和。例…

快速排序(hoare版本)

文章目录 文章目录 前言 二、使用步骤 1.实现基准值 2.递归实现排序 3.三数取中 三.注意事项 总结 前言 我们之前学习的多种排序&#xff0c;它们都有着不同的效率&#xff0c;可以适用与不同的场景&#xff0c;接下来要说的一种排序它叫做快速排序&#xff0c;从它的名字就可…

从产品经理到AI产品经理,这波升职加薪我把握住了

2024年&#xff0c;还有什么新风口&#xff1f; AI、元宇宙、NFT… 很多人不知道&#xff0c;其实不管是元宇宙还是NFT&#xff0c;它们本质上就是人工智能领域。 AI自身应用领域非常广泛&#xff0c;大批高薪岗位随之涌了出来&#xff0c;包括AI产品经理。 AI产品经历具体工…

微软运用欺骗性策略大规模打击网络钓鱼活动

微软正在利用欺骗性策略来打击网络钓鱼行为者&#xff0c;方法是通过访问 Azure 生成外形逼真的蜜罐租户&#xff0c;引诱网络犯罪分子进入以收集有关他们的情报。 利用收集到的数据&#xff0c;微软可以绘制恶意基础设施地图&#xff0c;深入了解复杂的网络钓鱼操作&#xff…

数字化工厂:制造业转型的新引擎

在当前技术飞速发展的时代,数字化正以前所未有的速度深入到各个行业,推动着产业转型升级。制造业作为国民经济的支柱,更是数字化转型的重点领域。随着5G、大数据、云计算、人工智能等新一代信息技术的广泛应用,以及国家"工业4.0"、"中国制造2025"等政策的持…

05-服务保护和分布式事务

原文为黑马程序员的飞书云文档&#xff0c;链接在这&#xff1a;原文链接 在微服务的远程调用中&#xff0c;还存在几个问题需要解决&#xff1a; 首先是业务健壮性问题&#xff1a; 在之前的查询购物车列表的业务中&#xff0c;购物车服务需要查询最新的商品信息&#xff0…