【问题标题】:Usage of spinlock functions自旋锁函数的使用
【发布时间】:2015-04-29 21:21:28
【问题描述】:

如何使用这些功能? 我已全局声明锁定。

pthread_spinlock_t lock;

自旋锁也在本地初始化。

pthread_spin_init(&lock, 1); // non-zero as pshared for IPC

但现在我想锁定我的关键整数并增加它。 我有多个进程循环运行:

while(0 != pthread_spin_trylock(&lock));
criticalInt += 1;
pthread_spin_unlock(&lock);

为什么这不起作用? 还有,下面这个函数是怎么用的?

pthread_spin_lock(&lock);

编辑:

for (i=0; i < NUM_CHILDREN; i++) {

   pid[i] = fork();
   if (pid[i] == -1) { return EXIT_FAILURE; }
   if (pid[i] == 0) {
   while (criticalInt < MAXCOUNT) {
      pthread_spin_lock(&lock);
      criticalInt += 1;
      pthread_spin_unlock(&lock);
      count++;
   }
   printf("Process %i counted %i\n", i, count);

}

对于 1000000 的 MAXCOUNT,帽子会产生以下输出:

    Process 3 counted 687858
    Process 0 counted 815657
    Process 1 counted 640191
    Process 2 counted 744340

实际上加起来应该是1000000。但他们不是。

如果我完全删除锁,我会得到类似的结果。

【问题讨论】:

  • 你能准确描述一下是什么行为让你说它不起作用吗?您收到错误消息吗?你的程序挂了吗?它会产生意外的输出吗?
  • 非常抱歉,我忘记了循环。好吧,它根本不会锁定我的应用程序。信号量就像一个魅力,但不是这个自旋锁。
  • 什么——你自己在纺纱?只需使用 pthread_spin_lock 而不是那个循环。您的代码的预编辑形式是正确的。
  • 另外,什么行为让你说它“根本不锁定我的应用程序”?
  • pthread_spin_lock 在获得锁之前不会返回。看起来这就是您试图通过循环 trylock 来实现的目标。

标签: c ipc spinlock


【解决方案1】:

要使 POSIX 线程的任何典型用法正常工作,您需要使用 pthread_create() 来启动一个新线程。在这里,您使用的是fork(),它启动了一个与父进程不同的地址空间的新进程。

更新:@Ramiz 在 cmets 中指出,一些 pthread 原语可以利用共享内存区域,从而允许它们跨分叉进程进行操作。这不是使用 pthread 东西的常用方法,但它是可能的!

【讨论】:

  • criticalInt 实际上是共享内存。
  • 共享内存中是否也“锁定”了?
  • 不,我怎么把它弄进去?
  • 如果您查看 pthread_spinlock_t 的 typedef,它只是“volatile int”。如果您可以像声明 criticalInt 一样声明锁,它可能会起作用。
  • 这个答案完全正确:您可以在分叉的进程上使用 pthread_spin_lock,但他们必须有权访问锁,例如通过共享内存和 pthread_spin_init 必须使用标志 PTHREAD_PROCESS_SHARED 完成,请参阅man7.org/linux/man-pages/man3/pthread_spin_init.3.html 但是,不建议推荐这种方法。
【解决方案2】:

如果您希望自旋锁被多个进程而不是线程使用,那么自旋锁本身也需要在共享进程内存中生成。 PTHREAD_PROCESS_SHARED 的 pshared 值本身不足以在进程之间共享自旋锁。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2011-10-19
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-09-27
    • 2016-08-12
    • 2020-02-02
    相关资源
    最近更新 更多