atomic_bool值更新,其他进程看不到

时间:2015-08-08 22:38:25

标签: c struct c11

我有一个程序有两个与共享内存通信的进程。在ctrl-c上我希望两个进程都退出。我使用名为stop的atomic_bool变量来通知进程在设置为true时保持循环或退出。但是,当atomic_bool变量stop设置为true时,其他进程不会看到更改。这意味着它仍然打印出0而不是1,但是进行更改的过程显示1.那么为什么第二个进程看不到从false变为true?

Control-c无法杀死该进程,因此请改用killall

#include <string.h>
#include <unistd.h>
#include <stdio.h>
#include <signal.h>
#include <stdbool.h>
#include <stdatomic.h>
#include <sys/mman.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <errno.h>

struct shared_map
{

    atomic_bool stop;

};

struct shared_map *map;

int compare_and_swap_loop(atomic_bool target, int value)
{
    /* Loop until we can succesfully update the the value. */
    while(1)
    {
        /* Grab a snapshot of the value that need to be updated. */
        bool snapshot = atomic_load(&target);

        if(atomic_compare_exchange_weak(&target, &snapshot, value) == true)
        {
            /* We succesfully updated the value let's exit this loop and return. */
            break;
        }
    }

   printf("result: %d\n", atomic_load(&target));

    return 0;
}

static void ctrlc_handler(int sig)
{
    compare_and_swap_loop(&map->stop, true);

    return;
}

void setup_signal_handler(void)
{
    (void) signal(SIGINT, ctrlc_handler);

    return;
}

static int create_shared(void **pointer, int size)
{
    *pointer = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_ANON | MAP_SHARED, -1, 0);
    if(*pointer == MAP_FAILED)
    {
        printf("mmap: %s\n", strerror(errno));
        return -1;
    }

    return 0;
}

static void loop(void)
{
    /* Set up signal handler. */
    setup_signal_handler();

    /* Check if we should stop or continue running. */
    while(atomic_load(&map->stop) == false)
    {
        sleep(2);

        printf("map->stop: %d\n", atomic_load(&map->stop));
    }

    return;
}

int main(void)
{
    int rtrn;
    pid_t pid;

    rtrn = create_shared((void **)&map, sizeof(struct shared_map));
    if(rtrn < 0)
    {
        printf("Can't create shared memory\n");
        return -1;
    }

    atomic_init(&map->stop, false);

    pid = fork();
    if(pid == 0)
    {
        loop();

        _exit(0);
    }
    else if(pid > 0)
    {
        int status;

        waitpid(pid, &status, 0);

        return 0;
    }
    else
    {
        printf("fork: %s\n", strerror(errno));
        return -1;
    }

    return 0;
}

2 个答案:

答案 0 :(得分:6)

您将原子变量的副本传递给compare_and_swap_loop函数,这对您没有任何好处 - 您需要处理在您的进程之间共享的相同值。

你需要这样做:

int compare_and_swap_loop(atomic_bool *target, int value)
{
   /* Loop until we can succesfully update the the value. */
   while(1)
   {
    /* Grab a snapshot of the value that need to be updated. */
       bool snapshot = atomic_load(target);

       if(atomic_compare_exchange_weak(target, &snapshot, value) == true)
       {
        /* We succesfully updated the value let's exit this loop and return. */
        break;
       }
   }

   printf("result: %d\n", atomic_load(target));

   return 0;
}

答案 1 :(得分:0)

以下代码的编写没有&#39; atomic _ *&#39;命令 但确实显示了您的流程的错误。

大多数情况下,需要杀死孩子的pid,而不是父pid

以下代码显示子pid,因此很容易找到

#define _GNU_SOURCE

#include <string.h>
#include <unistd.h>
#include <stdio.h>
#include <signal.h>
#include <stdbool.h>
//#include <stdatomic.h>
#include <sys/mman.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <errno.h>

struct shared_map
{
    bool stop;
};

struct shared_map *map;

int compare_and_swap_loop( bool *target )
{
    /* Loop until we can succesfully update the the value. */
    while(1)
    {
        /* Grab a snapshot of the value that need to be updated. */
        bool snapshot = *target;

        if(snapshot)
        {
            /* We succesfully updated the value let's exit this loop and return. */
            break;
        }
    }

    return 0;
}

static void ctrlc_handler(int sig)
{
    if( SIGINT==sig)
    compare_and_swap_loop(&map->stop);

    return;
}

void setup_signal_handler(void)
{
    (void) signal(SIGINT, ctrlc_handler);

    return;
}

static int create_shared(void **pointer, size_t size)
{
    *pointer = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_ANON | MAP_SHARED, -1, 0);
    if(*pointer == MAP_FAILED)
    {
        perror("mmap failed");
        return -1;
    }

    return 0;
}

static void loop(void)
{
    /* Set up signal handler. */
    setup_signal_handler();

    /* Check if we should stop or continue running. */
    while(!map->stop)
    {
        sleep(2);

        printf("map->stop: %d\n", map->stop);
    }

    return;
}

int main(void)
{
    int rtrn;
    pid_t pid;

    printf( "entered Main\n");

    rtrn = create_shared((void **)&map, sizeof(struct shared_map));
    if(rtrn < 0)
    {
        printf("Can't create shared memory\n");
        return -1;
    }

    map->stop = false;

    pid = fork();
    if(pid == 0)
    { //then child
        printf( "child process\n");
        loop();

        _exit(0);
    }
    else if(pid > 0)
    { // then parent
        int status;
        printf( "parent process\n");
        printf( "child Pid: %d\n", pid);

        waitpid(pid, &status, 0);

        return 0;
    }
    else
    { // else, fork failed
        perror("fork failed");
        return -1;
    }

    return 0;
}