多线程同步机制
发布日期:2022-04-11 08:53:00 浏览次数:3 分类:博客文章

本文共 5512 字,大约阅读时间需要 18 分钟。

1, spinlock

适合短暂的等待,不值得休眠的那种短暂.

jmp 11b12:get_lock:    lock bts $0, 0x6000    jc get_lock  lock incw smp_cpus  mov $smp_cpus, %bx  lock btr $0, 0x6000    /*release lock*/  mov 0(%bx), %si  ...smp_ap_boot_code_end:

这是每个核启动后,根据smp_cpus获取自己的核编号,从而在屏幕上定位不同的"点",然后循环递增其ascii码.

 

2, atomic_t

这是linux内核提供的原子数,相应的有atomic_inc/dec()一系列方法.

x86上,其内部实现依赖于lock指令前缀.

 

3,semaphore

sem_init(); sem_wait(); sem_post(); sem_destroy();

如果想fork()后仍然使用,需要利用共享内存区.

另外每次post之后,只唤醒队列里的一个进程.因为等待队列里的进程肯定都是因为sem_wait()进入的,他们自然都会调用sem_post()叫"下一个".

SO上的一篇特别好:

 

I want to fork multiple processes and then use a semaphore on them. Here is what I tried:

sem_init(&sem, 1, 1); /* semaphore*, pshared, value */...if(pid != 0){ /* parent process */ wait(NULL); /* wait all child processes */ printf("\nParent: All children have exited.\n"); . . /* cleanup semaphores */ sem_destroy(&sem); exit(0);}else{ /* child process */ sem_wait(&sem); /* P operation */ printf(" Child(%d) is in critical section.\n",i); sleep(1); *p += i%3; /* increment *p by 0, 1 or 2 based on i */ printf(" Child(%d) new value of *p=%d.\n",i,*p); sem_post(&sem); /* V operation */ exit(0);} .....
 
一个高赞回答:

The problem you are facing is the misunderstanding of sem_init() function. When you read the you will see this:

The pshared argument indicates whether this semaphore is to be shared between the threads of a process, or between processes.

If you are done reading up to this point, you will think that the non-zero value of pshared will make the semaphore inter-process semaphore. However, this is wrong. You should continue reading and you'll understand that you have to locate the semaphore in a shared memory region. To do that, several functions can be used as you can see below:

If pshared is nonzero, then the semaphore is shared between processes, and should be located in a region of shared memory (see shm_open(3), mmap(2), and shmget(2)). (Since a child created by fork(2) inherits its parent's memory mappings, it can also access the semaphore.) Any process that can access the shared memory region can operate on the semaphore using sem_post(3), sem_wait(3), etc.

I find this approach as a more complicated approach than others, therefore I want to encourage people to use sem_open() instead of sem_init().

Below you can see a complete program illustrates the following:

  • How to allocate shared memory and use shared variables between forked processes.
  • How to initialize a semaphore in a shared memory region and is used by multiple processes.
  • How to fork multiple processes and make the parent wait until all of its children exit.
#include 
/* printf() */#include
/* exit(), malloc(), free() */#include
/* key_t, sem_t, pid_t */#include
/* shmat(), IPC_RMID */#include
/* errno, ECHILD */#include
/* sem_open(), sem_destroy(), sem_wait().. */#include
/* O_CREAT, O_EXEC */int main (int argc, char **argv){ int i; /* loop variables */ key_t shmkey; /* shared memory key */ int shmid; /* shared memory id */ sem_t *sem; /* synch semaphore *//*shared */ pid_t pid; /* fork pid */ int *p; /* shared variable *//*shared */ unsigned int n; /* fork count */ unsigned int value; /* semaphore value */ /* initialize a shared variable in shared memory */ shmkey = ftok ("/dev/null", 5); /* valid directory name and a number */ printf ("shmkey for p = %d\n", shmkey); shmid = shmget (shmkey, sizeof (int), 0644 | IPC_CREAT); if (shmid < 0){ /* shared memory error check */ perror ("shmget\n"); exit (1); } p = (int *) shmat (shmid, NULL, 0); /* attach p to shared memory */ *p = 0; printf ("p=%d is allocated in shared memory.\n\n", *p); /********************************************************/ printf ("How many children do you want to fork?\n"); printf ("Fork count: "); scanf ("%u", &n); printf ("What do you want the semaphore value to be?\n"); printf ("Semaphore value: "); scanf ("%u", &value); /* initialize semaphores for shared processes */ sem = sem_open ("pSem", O_CREAT | O_EXCL, 0644, value); /* name of semaphore is "pSem", semaphore is reached using this name */ printf ("semaphores initialized.\n\n"); /* fork child processes */ for (i = 0; i < n; i++){ pid = fork (); if (pid < 0) { /* check for error */ sem_unlink ("pSem"); sem_close(sem); /* unlink prevents the semaphore existing forever */ /* if a crash occurs during the execution */ printf ("Fork error.\n"); } else if (pid == 0) break; /* child processes */ } /******************************************************/ /****************** PARENT PROCESS ****************/ /******************************************************/ if (pid != 0){ /* wait for all children to exit */ while (pid = waitpid (-1, NULL, 0)){ if (errno == ECHILD) break; } printf ("\nParent: All children have exited.\n"); /* shared memory detach */ shmdt (p); shmctl (shmid, IPC_RMID, 0); /* cleanup semaphores */ sem_unlink ("pSem"); sem_close(sem); /* unlink prevents the semaphore existing forever */ /* if a crash occurs during the execution */ exit (0); } /******************************************************/ /****************** CHILD PROCESS *****************/ /******************************************************/ else{ sem_wait (sem); /* P operation */ printf (" Child(%d) is in critical section.\n", i); sleep (1); *p += i % 3; /* increment *p by 0, 1 or 2 based on i */ printf (" Child(%d) new value of *p=%d.\n", i, *p); sem_post (sem); /* V operation */ exit (0); }}
 

另外关注sem_open()这个函数.

转载地址:https://www.cnblogs.com/xiang-yin/p/12114209.html 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!

上一篇:多线程同步的方法
下一篇:多线程同步控制 ManualResetEvent AutoResetEvent MSDN

发表评论

最新留言

路过按个爪印,很不错,赞一个!
[***.219.124.196]2024年03月31日 14时00分32秒