On Mon, Mar 10, 2025 at 11:49:39PM +0100, Aurelien Jarno wrote:
> I also noticed it failed the same way on arm64.

Thanks for confirming!

> > > Please tell me if you would like me to do more experiments.
> > 
> > If you have time, yes, that's very useful. I'll also try to have a
> > closer look at your testcase, but it already seems relatively small.
> 
> I am not sure I understand everything yet, but it seems that building
> pari with -fexceptions is a good workaround for the issue.

Alas, this would break the ABI of the libpari,
and I still get failure with -fexceptions: instead of crashes, the program
terminate without printing Done:

%for i in `seq 1 100`; do (set -e; echo -n "$i " && sid ./pthread1) || break; 
done
1 Done!
2 3 4 5 6 Done!
7 8 Done!
9 10 11 12 13 Done!
14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 Done!
29 30 31 Done!
32 33 34 Done!
35 36 37 38 39 40 Done!
41 42 43 44 45 46 Done!
47 48 49 50 51 52 53 Done!
54 55 56 57 58 59 60 61 62 63 64 65 ^C%

I join a simplified test-case.

Cheers,
-- 
Bill. <ballo...@debian.org>

Imagine a large red swirl here. 
/* Copyright (C) 2013  The PARI group.

This file is part of the PARI/GP package.

PARI/GP is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation; either version 2 of the License, or (at your option) any later
version. It is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY WHATSOEVER.

Check the License for details. You should have received a copy of it, along
with the package; see the file 'COPYING'. If not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

struct mt_state
{
  long workid;
};

struct mt_queue
{
  long workid;
};

struct mt_pstate
{
  pthread_t *th;
  void **pth;
  struct mt_queue *mq;
  long n;
};

static void
mt_queue_cleanup(void *arg)
{
  (void) arg;
}

static void*
mt_queue_run(void *arg)
{
  pthread_cleanup_push(mt_queue_cleanup,NULL);
  for(;;)
  {
    pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS,NULL);
    for(long u=1; u<10000; u++);
    pthread_setcanceltype(PTHREAD_CANCEL_DEFERRED,NULL);
  }
  pthread_cleanup_pop(1);
  return NULL;
}

void
mt_queue_start_lim(long lim)
{
  struct mt_pstate *mt =
         (struct mt_pstate*) malloc(sizeof(struct mt_pstate));
  long i;
  mt->mq  = (struct mt_queue *) malloc(sizeof(*mt->mq)*lim);
  mt->th  = (pthread_t *) malloc(sizeof(*mt->th)*lim);
  mt->pth = (void **) malloc(sizeof(*mt->pth)*lim);
  mt->n = lim;
  for (i=0;i<lim;i++)
  {
    struct mt_queue *mq = mt->mq+i;
    mt->pth[i] = (void*)mq;
  }
  for (i=0;i<lim;i++)
    pthread_create(&mt->th[i],NULL, &mt_queue_run, (void*)mt->pth[i]);
  for (i=0; i<lim; i++)
    pthread_cancel(mt->th[i]);
  for (i=0; i<lim; i++)
    pthread_join(mt->th[i],NULL);
}

int
main(void)
{
  long k;
  for (k = 1; k<1000; k++)
    mt_queue_start_lim(8);
  printf("Done!\n");
}

Reply via email to