虚拟机串口与主机串口通信·小程序(下)

上次说到的,不能做到实时通信。那么开两个进程就可以了,一个用来监听是否有消息传来,一个用来等待用户输入。那么,先来复习一下进程的相关概念。
<br>

进程结构

linux中进程包含PCB(进程控制块)、程序以及程序所操纵的数据结构集,可分为“代码段”、“数据段”和“堆栈段”。

进程状态

运行状态R(TASK_RUNNING)

可中断睡眠状态S(TASK_INTERRUPTIBLE)
不可中断睡眠状态D(TASK_UNINTERRUPTIBLE)

暂停状态T(TASK_STOPPED或TASK_TRACED)

僵死状态Z(TASK_ZOMBIE)
退出状态X(TASK_DEAD)
ps -aux 查看进程信息,可以看到各进程的状态:
虚拟机串口与主机串口通信·小程序(下)

init进程

所有进程的父进程
init进程绝不会终止。

它是一个普通的用户进程(与交换进程不同,它不是内核中的系统进程),但是它以超级用户特权运行。

获取进程标识

#include < sys/types.h>
#include < unistd.h>
pid_t getpid(void); 返回:调用进程的进程I D
pid_t getppid(void); 返回:调用进程的父进程I D

uid_t getuid(void); 返回:调用进程的实际用户I D
uid_t geteuid(void); 返回:调用进程的有效用户I D
gid_t getgid(void); 返回:调用进程的实际组I D
gid_t getegid(void); 返回:调用进程的有效组I D

fork系统调用

#include < sys/types.h>
#include < unistd.h>
    pid_t fork(void);
    返回:子进程中为0,父进程中为子进程I D,出错为-1

注意:
<br>1、使用fork函数得到的子进程从父进程的继承了整个进程的地址空间,包括:
进程上下文、进程堆栈、内存信息、打开的文件描述符、信号控制设置、进程优先级、进程组号、当前工作目录、根目录、资源限制、控制终端等。

2、子进程与父进程的区别在于:
1)父进程设置的锁,子进程不继承
2)各自的进程ID和父进程ID不同
3)子进程的未决告警被清除;
4)子进程的未决信号集设置为空集。

3、fork系统调用之后,父子进程将交替执行。
4、如果父进程先退出,子进程还没退出。那么子进程的父进程将变为init进程。(注:任何一个进程都必须有父进程)
5、如果子进程先退出,父进程还没退出,那么子进程必须等到父进程捕获到了子进程的退出状态才真正结束,否则这个时候子进程就成为僵进程。(这是不好的,这样子进程会一直占用内存资源)
解决办法
1)wait函数:使父进程阻塞,直到一个子进程结束或者该进程收到一个指定信号为止。(但是,这样不好)
2)signal(SIGCHLD,SIG_IGN)。表示父进程忽略SIGCHLD,该信号是子进程退出的时候向父进程发出的,由init进程收。

测试

接下来,可以继续我们上次的小程序了。

#include < stdio.h>
#include < sys/types.h>
#include < signal.h>
#include " port.h"
#include " readConfig.h"

// 1、主机与虚拟机 
int main()
{
    int fd,pid,child_id;
    char filename[20] = "serial.cfg";
    char recbuf[100] = "";
    char sendbuf[100] = "";

    struct t_port port = {0};

    //从文件获取配置信息
    getMsg(filename,&port);
    //串口初始化
    fd = portInit(port.devname,port.speed,port.data,port.parity,port.stop);

    //主机与虚拟机串口通信
    pid = fork();
    if(pid == -1)
    {
        perror("create process");
        return 1;
    }

    else if(pid == 0) //子进程 接收消息
    {
        child_id = getpid();
        while(read(fd,recbuf,100)>0)  //这里不能用strlen。因为开始strlen(recbuf)=0
        {
            if(strcmp(recbuf,"") != 0)
            {
                printf("receive msg: %s\n",recbuf);
                putchar('$');
                fflush(stdout);
            }
            memset(recbuf,0,100);
        }

    }
    else    //(主)父进程 发送消息
    {
        printf("$");
        while(1)
        {
            memset(sendbuf,0,100);
            scanf("%s",sendbuf);
            if(strcmp(sendbuf,"over") == 0) //但是,此时,子进程未结束,还在后台运行
            {
                break;
            }
            if(write(fd,sendbuf,strlen(sendbuf)) > 0)
            {
                printf("$");
            }
        }

    }

    close(fd);
    return 0;
}

//2、虚拟机两串口
/*
int main(int argc,char *argv[])
{
    int fd,pid;
    char receivebuf[100] = "";
    char sendbuf[100] = "";

    if(argc != 2)
    {
        printf("format error!\n");
        return 1;
    }

    if( strcmp(argv[1],"ttyS0")==0)
    {
        fd = portInit("/dev/ttyS0",115200,8,0,1);
    }
    else if(strcmp(argv[1],"ttyS1")==0)
    {
        fd = portInit("/dev/ttyS1",115200,8,0,1);
    }
    else 
    {
        printf("format error!\n");
        return 1;
    }

    pid = fork();

    if(pid == -1)
    {
        perror("create process");
        return 1;
    }
    else if(pid == 0) //子进程
    {
        //接收消息
        while(read(fd,receivebuf,sizeof(receivebuf)) > 0)
        {
            if(strcmp(receivebuf,"") != 0)
            {
                printf("receive msg : %s\n",receivebuf);
            }
            memset(receivebuf,0,100);
        }

    }

    else    //父进程
    {
        //发送消息
        while(1)
        {
            memset(sendbuf,0,100);
            scanf("%s",sendbuf);
            if(strcmp(sendbuf,"-over") == 0)
            {
                break;
            }
            write(fd,sendbuf,strlen(sendbuf));
            printf("send OK\n");
        }
    }

    close(fd);
    return 0;
}
*/

运行结果:
虚拟机串口与主机串口通信·小程序(下)
<br>发现问题了没?对的。父进程会先死。
程序运行时:
虚拟机串口与主机串口通信·小程序(下)
程序结束时:
虚拟机串口与主机串口通信·小程序(下)

如何解决呢?下次说~

转载于:https://blog.51cto.com/13097817/2049588

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
串口通信C程序 //Set CommTimeOuts BOOL SetCommTimeOuts ( VOID ) { //SetComm Input & Output Buffer SetupComm ( hSerial ,MAXLEN * 2 ,MAXLEN * 2 ); //Read TimeOut TimeOuts.ReadIntervalTimeout = MAXDWORD; //Set Total Read TimeOut as MAXDWORD (0xFFFFFFFF) TimeOuts.ReadTotalTimeoutConstant = 0; //Read TimeOut Const TimeOuts.ReadTotalTimeoutMultiplier = 0; //Return Riht now //Write TimeOut TimeOuts.WriteTotalTimeoutMultiplier = 50; //TotalTimeOut = TimeAgr*NumOfChars+const time TimeOuts.WriteTotalTimeoutConstant = 2000; //Set Write TimeOuts return ( FALSE != SetCommTimeouts ( hSerial,&TimeOuts ) ); } //Error MsgBox BOOL ErrMsg ( HWND hWnd,TCHAR *szErr ) { return ( FALSE != MessageBox ( hWnd,szErr,NULL,MB_OK | MB_ICONWARNING ) ); } //Read Comm BOOL ReadCom ( HWND hSet,BYTE InBuff[],INT BytesNum ) { DWORD dwBytesRead = 0; //Record The bytes already read INT iBytesToRead = 0; DWORD dwErrMask = 0; //Clear Memory ZeroMemory ( &ComState ,sizeof(ComState) ); ZeroMemory ( &OvLap ,sizeof(OvLap) ); OvLap.Offset = 0; OvLap.OffsetHigh = 0; OvLap.hEvent = CreateEvent (NULL,TRUE,FALSE,NULL); //Clear All sPort error ClearCommError (hSerial,&dwErrMask,&ComState); //Get The Bytes to read from buff iBytesToRead = min (2000,ComState.cbInQue); if ( 0 == iBytesToRead ) return FALSE; else; if ( FALSE == ReadFile (hSerial,InBuff,iBytesToRead,&dwBytesRead,&OvLap ) ) { if ( GetLastError () == ERROR_IO_PENDING ) { WaitForSingleObject (OvLap.hEvent,2000); PurgeComm (hSerial,PURGE_TXABORT | PURGE_TXCLEAR | PURGE_RXCLEAR ); } else; } else; return (TRUE); } ///Write Comm BOOL WriteCom ( BYTE *szBuff,DWORD dwBytes ) { DWORD dwErrMask = 0 ; DWORD dwBytesWritten = 0; ZeroMemory ( &OvLap ,sizeof(OvLap) ); ZeroMemory ( &ComState,sizeof(ComState) ); OvLap.Offset = 0; OvLap.OffsetHigh = 0; OvLap.hEvent = CreateEvent (NULL,TRUE,FALSE,NULL); //Clear All sPort error ClearCommError (hSerial,&dwErrMask,&ComState); if ( FALSE == WriteFile ( hSerial,szBuff,dwBytes,&dwBytesWritten ,&OvLap ) ) { if ( GetLastError () == ERROR_IO_PENDING ) { WaitForSingleObject (OvLap.hEvent,2000) ; PurgeComm (hSerial,PURGE_TXABORT | PURGE_TXCLEAR | PURGE_RXCLEAR ); } else; } else; return ( dwBytesWritten == dwBytes ); } TCHAR* Caps ( TCHAR *szStr ) { UINT i = 0; for ( i = 0 ; szStr[i] != '\0'; i ++ ) { if ( szStr[i] >= 'A' && szStr[i] <= 'Z' ) szStr[i] += 0x20; } return szStr; }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值