Linux下使用C语言的信号灯实现进程同步的方法 (linux c 信号灯)
在多进程或多线程的程序中,进程的同步问题是非常关键的,一般通过信号灯的方式来实现,Linux下也有类似的API可以使用。本文将介绍如何在Linux下使用C语言的信号灯来实现进程同步的方法。
一、信号灯的基本概念
信号灯是一种同步机制,主要用于进程间的同步和互斥。信号灯有两个基本操作:P操作和V操作。
1. P操作(wt)
P操作用于实现进程互斥。当一个进程需要访问某一共享资源时,需要通过P操作获取该资源的信号灯,此时信号灯减1。如果信号灯的值为0,则表示资源被占用,该进程需要等待其他进程释放资源。
2. V操作(signal)
V操作用于实现进程同步。当一个进程完成对某一共享资源的访问后,需要通过V操作释放该资源的信号灯,此时信号灯加1。如果有其他进程正在等待该信号灯,则该信号灯将被唤醒。
二、信号灯API介绍
在Linux下,IPC机制提供了很多进程间通信的方式,其中包括信号灯。下面是Linux下信号灯API的介绍:
1. key_t ftok(const char *pathname, int proj_id);
函数用于获取一个唯一的key值。
2. int semget(key_t key, int nsems, int sem);
函数创建或获取一个信号灯的ID。
3. int semctl(int semid, int semnum, int cmd, union semun arg);
函数用于控制和获取信号灯的状态。
4. int semop(int semid, struct sembuf *sops, unsigned nsops);
函数用于进行信号灯的P操作和V操作。
五、示例程序
下面是一个示例程序,通过信号灯实现两个进程的同步。
1.父进程通过信号灯初始化信号灯的值,并且等待子进程执行完毕后释放资源。
2.子进程执行对共享资源的访问,并通过V操作释放信号灯。
代码如下:
“`c
#include
#include
#include
#include
#include
#include
//信号灯union
union semun {
int val; /* Value for SETVAL */
struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */
unsigned short *array; /* Array for GETALL, SETALL */
struct seminfo *__buf; /* Buffer for IPC_INFO (Linux-specific) */
};
int mn()
{
int semid;
int pid;
union semun arg;
struct sembuf buf;
//获取key
key_t key = ftok(“./”, 123);
//创建信号灯
semid = semget(key, 1, IPC_CREAT | 0666);
//初始化信号灯的值
arg.val = 0;
if(semctl(semid, 0, SETVAL, arg) == -1) {
perror(“Fled to initialize semaphore”);
exit(EXIT_FLURE);
}
//创建子进程
pid = fork();
if(pid == -1) {
perror(“Fled to fork”);
exit(EXIT_FLURE);
}
if(pid == 0) {
//子进程执行对共享资源的访问,并通过V操作释放信号灯
sleep(2);
printf(“Child process has accessed the shared resources.\n”);
buf.sem_num = 0;
buf.sem_op = 1;
buf.sem_ = 0;
if(semop(semid, &buf, 1) == -1) {
perror(“Fled to semop”);
exit(EXIT_FLURE);
}
exit(EXIT_SUCCESS);
} else {
//父进程等待子进程执行完毕后释放信号灯
buf.sem_num = 0;
buf.sem_op = -1;
buf.sem_ = 0;
if(semop(semid, &buf, 1) == -1) {
perror(“Fled to semop”);
exit(EXIT_FLURE);
}
wt(NULL);
printf(“Parent process has released the shared resources.\n”);
}
return 0;
}
“`
三、
相关问题拓展阅读:
- 嵌入式C语言重点知识点
嵌入式C语言重点知识点
嵌入式C语言重点知识点
嵌入式LINUX
嵌入式Linux 是将日益流行的Linux操作系统进行裁剪修改,使之能在嵌入式计算机系统上运行的一种操作系统。Linux做嵌入式的优势,首先,Linux是开放源代码;其次,Linux的内核小、效率高,可以定制,其系统内核最小只有约134KB;第三,Linux是免费的OS,Linux还有着嵌入式操作系统所需要的很神神多特色,突出的就是Linux适应于多种CPU和多种硬件平台而且性能稳定,裁剪性很好,开发和使用都很容易。同时,Linux内核的结构在网络方面是非常完整的,Linux对网络中最常用的TCP/IP协议有最完备的支持。提供了包括十兆、百兆、千兆的以太网络,以及无线网络,Token Ring(令牌环网)、光纤甚至卫星的支持。
移植步骤:
1.Bootloader的移植;
2.嵌入式Linux操作系统内核的移植;
3.嵌入式Linux操作系统根文件系统的创建;
4.电路板上外设Linux驱动程序的编写。
WinCE
WinCE是微软公司嵌入式、移动计算平台的基础,它是一个开放的、可升级的32位嵌入式操作系统,是基于掌上型电脑类的电子设备操作系统,它是精简的Windows 95,Win CE的图形用户界面相当出色。WinCE是从整体上为有限资源的平台设计的多线程、完整优先权、多任务的操作系统。它的模块化设计允许它对于从掌上电脑到专用的工业控制器的用户电子设备进行定制。操作系统的基本内核需要至少200K的ROM。
一般来说,一个WinCE系统包括四层结构:应用程序、WinCE内核映像、板级支持包(BSP)、硬件平台。而基本软件平台则主要由WinCE系统内核映像(OS Image)和板卡支持包(BSP)两部分组成。因为WinCE系统是一个软硬件紧密结合的系统,因此即使CPU处理器相同,但是如果开发板上的外围硬件不相同,这个时候还是需要修改BSP来完成一个新的BSP。因此换句话说,就是WinCE的移植过程主要是改写BSP的过程。
Android
Android 是一个包括操作系统,中间件以及一些重要应用程序的专门针对移动设备的层次结构的软件集。Android 作为一个完全开源的.操作系统,是由操作系统Linux、中间件以及核心应用程序组成的软件栈。通过 android SDK 提供的 API 以及相应的开发工具, 程序员可以很方便的开发android平台上的应用程序。其整个系统由应用程序,应用程序框架,应用程序库,Android运行库,Linux内核(Linux Kernel)五个部分组成。Android操作系统内置了一部分应用程序, 包括电子邮件客户端、S程序、日历、地图、浏览器、通讯录以及其他的程序,值得一提的是这些所有的程序都是用java编写的。
银山 移植的主要的工作是驱动,硬件抽象层的移植。为了更好地理解和调试系统,也应该适当地了解上层对硬件抽象层的调用情况。
TinyOS
TinyOS是一个开源的嵌入式操作系统,它是由加州大学的伯利克分校开发出来的,主要应用于无线传感器网络方面。程序采用的是模块化设计,所以它的程序核心往往都很小,一般来说核心代码和数据大概在400 Bytes左右,能够突破传感器存储资源少的限制。TinyOS提供一系列可重用的组件,一个应用锋瞎中程序可以通过连接配置文件(A Wiring Specification)将各种组件连接起来,以完成它所需要的功能。
嵌入式实时操作系统(RTOS)
在工业控制、 军事设备、航空航天等领域对系统的响应时间有苛刻的要求,这就需要使用实时系统。当外界事件或数据产生时,能够接受并以足够快的速度予以处理,其处理的结果又能在规定的时间之内来控制生产过程或对处理系统作出快速响应,并控制所有实时任务协调一致运行的嵌入式操作系统。故对嵌入式实时操作系统的理解应该建立在对嵌入式系统的理解之上加入对响应时间的要求。
FreeRTOS
FreeRTOS是一个迷你操作系统内核的小型嵌入式系统。作为一个轻量级的操作系统,功能包括:任务管理、时间管理、信号量、消息队列、内存管理、记录功能等,可基本满足较小系统的需要。FreeRTOS任务可选择是否共享堆栈,并且没有任务数限制,多个任务可以分配相同的优先权。相同优先级任务的轮转调度,同时可设成可剥夺内核或不可剥夺内核。
FreeRTOS 的移植主要需要改写如下三个文件。
1.portmacro.h
2.port.c
3. port.a
μTenux
μTenux基于ARM微控制器平台,对uT最适用于ARM Cortex M0-M4系列的微控制器,代码开源、免费,是一个功能强大的抢占式实时多任务操作系统。μTenux除具有实时嵌入式操作系统的一般特性:可移植性,可固化,可裁剪等特性以外,它还具有如下优点:
(1)微内核。无MMU, ROM/RAM占用量小,所占ROM更大60KB,最小10KB;RAM更大12KB,最小2KB;
(2)开源免费;
(3)支持所有32位ARM7/9和Cortex M系列的微控制器;
(4)可配置多达到256个任务以及140个任务优先级;
(5)有良好的商业支持, T-Engine论坛进行总的维护。
移植主要包括:芯片系统时钟移植,外设移植和通用输出/输入端口的移植以及看门狗模块移植。由于考虑到内核代码的重要性以及其在整个移植中的重要意义,且为了整个系统有更好的实时性,可选用汇编语言编写操作系统的启动代码。
VxWorks
VxWorks系统提供多处理器间和任务间高效的信号灯、消息队列、管道、网络透明的套接字。实时系统的另一关键特性是硬件中断处理。为了获得最快速可靠的中断响应,VxWorks系统的中断服务程序ISR有自己的上下文。VxWorks实时操作系统由400多个相对独立的、短小精炼的目标模块组成,用户可根据需要选择适当模块来裁剪和配置系统,这有效地保证了系统的安全性和可靠性。系统的链接器可按应用的需要自动链接一些目标模块。这样,通过目标模块之间的按需组合,可得到许多满足功能需求的应用。
移植过程可以参考网络上一些BSP代码,BSP的英文全称为board support package,即板级支持包,它的作用是针对特殊的硬件平台,为VxWorks内核提供操作的接口。
μClinux
嵌入式Linux作为一个开放源代码的操作系统,以价格低廉、功能强大又易移植的特性正在被广泛应用,μClinux是专门针对没有MMU的处理器而设计的嵌入式Linux,非常适合中低端嵌入式系统的需求。 在GNU通用公共许可证的授权下,μClinux操作系统的用户可以使用几乎所有Linux的API函数,不会因为没有内存管理单元MMU而受到影响;而且,μClinux在标准的Linux基础上进行了适当的裁剪和优化,形成了一个高度优化的、代码紧凑的嵌入式Linux,体积小了,但是仍然保留了Linux的大多数的优点,比如稳定性好、强大的网络功能、良好的可移植性、完备的文件系统支持功能、以及标准丰富的应用程序接口API等,可以支持类似ARM7TDMI等类型多的小巧玲珑的中央处理器。
eCos
eCos中文翻译为嵌入式可配置操作系统或嵌入式可配置实时操作系统。适合于深度嵌入式应用,主要应用对象包括消费电子、电信、车载设备、手持设备以及其他一些低成本和便携式应用。eCos是一种开发源代码软件,无任何版权费用。 eCos更大的特点是模块化,内核可配置。如果说嵌入式Linux太庞大了,那么eCos可能就能够满足要求。它是一个针对16位、32位和64位处理器的可移植开放源代码的嵌入式RTOS。和嵌入式Linux不同,它是由专门设计嵌入式系统的工作组设计的。eCos具有相当丰富的特性和一个配置工具,后者能够让你选取你所需要的特性。
eCos的软件分了若干的模块,移植工作主要在他的hal层进行,所谓hal(硬件抽象层)就是把和硬件相关的软件凑到一起。
μC/OS-II
μC/OS-II是一个完整的、可移植、可固化、可裁剪的占先式实时多任务内核。μC/OS-II绝大部分的代码是用ANSI的C语言编写的,包含一小部分汇编代码,使之可供不同架构的微处理器使用。其结构小巧简洁且支持抢占式的多任务调度与管理。此实时操作系统管理任务数多达64个,且提供内部程序存储器管理、系统运行时间管理、多任务实时调度与管理等功能。由于它的作者占用和保留了8个任务,所以留给用户应用程序最多可有56个任务。赋予各个任务的优先级必须是不相同的。这意味着μC/OS-II不支持时间片轮转调度法。μC/OS-II为每个任务设置独立的堆栈空间,可以快速实现任务切换。
将μC/OS-II操作系统移植到目标处理器上,需要从硬件和软件两方面来考虑。硬件方面,目标处理器需满足以下条件:
①处理器的C编译器能产生可重入代码;
②用C语言可以开/关中断;
③处理器支持中断,并且能够产生定时中断(通常在10~1000 Hz之间);
④处理器能够支持容纳一定量数据的硬件堆栈;
⑤处理器有将堆栈指针和其他寄存器读出和存储到堆栈或内存中的指令。
软件方面,主要是一些与处理器相关的代码移植,其分布在OS_CPU.H、OS_CPU_C.C和OS_CPU_A.A这3个不同的文件中。
关于linux c 信号灯的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。