X-Recipient: archive-cygwin AT delorie DOT com DKIM-Filter: OpenDKIM Filter v2.11.0 sourceware.org 9D3823841921 DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=cygwin.com; s=default; t=1699891999; bh=0tXN/KyP9M2P3MfNVswBJobxbyrmMHue0dyfyglms/8=; h=Date:To:Subject:References:In-Reply-To:List-Id:List-Unsubscribe: List-Archive:List-Post:List-Help:List-Subscribe:From:Reply-To:Cc: From; b=an+a6phJGWyxoEvuuyVvLy5/DRbvcaRuxNCwWq20bX1Cny25ptAWI5Pt3zReK2kYj xPGpuT3uWjmgCwzTC/9WeEzOXkB7A9KIuKUM2ACcaSZncEgjWrSly7CeEkhkEyOyDw /fsVUGyS15q6QXdVWEPijduK5eVSlbd6G4Lvx73M= X-Original-To: cygwin AT cygwin DOT com Delivered-To: cygwin AT cygwin DOT com DKIM-Filter: OpenDKIM Filter v2.11.0 sourceware.org 7B3BC3857C43 Date: Mon, 13 Nov 2023 17:12:39 +0100 To: Bruno Haible Subject: Re: random is not multithread-safe in Cygwin Message-ID: Mail-Followup-To: Bruno Haible , cygwin AT cygwin DOT com References: <3811044 DOT 57xzQst1vy AT nimes> MIME-Version: 1.0 Content-Disposition: inline In-Reply-To: <3811044.57xzQst1vy@nimes> X-BeenThere: cygwin AT cygwin DOT com X-Mailman-Version: 2.1.30 Precedence: list List-Id: General Cygwin discussions and problem reports List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , From: Corinna Vinschen via Cygwin Reply-To: cygwin AT cygwin DOT com Cc: Corinna Vinschen , cygwin AT cygwin DOT com Content-Type: text/plain; charset="us-ascii" Content-Transfer-Encoding: 7bit Errors-To: cygwin-bounces+archive-cygwin=delorie DOT com AT cygwin DOT com Sender: "Cygwin" Hi Bruno, On Nov 10 17:19, Bruno Haible via Cygwin wrote: > The function 'random' is, unlike 'rand', not marked as not MT-safe in POSIX > [1][2]. Thus it must be multithread-safe [3]: > "Each function defined in the System Interfaces volume of POSIX.1-2017 > is thread-safe unless explicitly stated otherwise." > > And indeed glibc, musl libc, AIX, Android, and even NetBSD implement it in a > multithread-safe way. Our code is from FreeBSD, originally. I checked the latest code from FreeBSD. It doesn't lock anything in random() and generates the same error when running the same test app. Why is that ok for FreeBSD? Corinna > > On Cygwin 2.9.0 and 3.4.6, it is not multithread-safe. > > How to reproduce: > 1. Compile the attached program. > $ x86_64-pc-cygwin-gcc foo.c > 2. Run it. > $ ./a.exe > Expected: No output. > Actual: Output such as > Expected value #367 not found in multithreaded results. > > [1] https://pubs.opengroup.org/onlinepubs/9699919799/functions/initstate.html > [2] https://pubs.opengroup.org/onlinepubs/9699919799/functions/rand.html > [3] https://pubs.opengroup.org/onlinepubs/9699919799/basedefs/V1_chap03.html#tag_03_407 > /* Multithread-safety test for random(). > Copyright (C) 2023 Free Software Foundation, Inc. > > This program 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 3 of the License, or > (at your option) any later version. > > This program is distributed in the hope that it will be useful, > but WITHOUT ANY WARRANTY; without even the implied warranty of > MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > GNU General Public License for more details. > > You should have received a copy of the GNU General Public License > along with this program. If not, see . */ > > /* Written by Bruno Haible , 2023. */ > > /* Whether to help the scheduler through explicit yield(). > Uncomment this to see if the operating system has a fair scheduler. */ > #define EXPLICIT_YIELD 1 > > /* Number of simultaneous threads. */ > #define THREAD_COUNT 4 > > /* Number of random() invocations operations performed in each thread. > This value is chosen so that the unit test terminates quickly. > To reliably determine whether a random() implementation is multithread-safe, > set REPEAT_COUNT to 1000000 and run the test 100 times: > $ for i in `seq 100`; do ./test-random-mt; done > */ > #define REPEAT_COUNT 1000000 > > /* Specification. */ > #include > > #include > #include > #include > > #if EXPLICIT_YIELD > # include > # define yield() sched_yield () > #else > # define yield() > #endif > > /* This test runs REPEAT_COUNT invocations of random() in each thread and stores > the result, then compares the first REPEAT_COUNT among these > THREAD_COUNT * REPEAT_COUNT > random numbers against a precomputed sequence with the same seed. */ > > static void * > random_invocator_thread (void *arg) > { > long *storage = (long *) arg; > int repeat; > > for (repeat = 0; repeat < REPEAT_COUNT; repeat++) > { > storage[repeat] = random (); > yield (); > } > > return NULL; > } > > int > main () > { > unsigned int seed = 19891109; > > /* First, get the expected sequence of random() results. */ > srandom (seed); > long *expected = (long *) malloc (REPEAT_COUNT * sizeof (long)); > assert (expected != NULL); > { > int repeat; > for (repeat = 0; repeat < REPEAT_COUNT; repeat++) > expected[repeat] = random (); > } > > /* Then, run REPEAT_COUNT invocations of random() each, in THREAD_COUNT > separate threads. */ > pthread_t threads[THREAD_COUNT]; > long *thread_results[THREAD_COUNT]; > srandom (seed); > { > int i; > for (i = 0; i < THREAD_COUNT; i++) > { > thread_results[i] = (long *) malloc (REPEAT_COUNT * sizeof (long)); > assert (thread_results[i] != NULL); > } > for (i = 0; i < THREAD_COUNT; i++) > assert (pthread_create (&threads[i], NULL, random_invocator_thread, thread_results[i]) == 0); > } > > /* Wait for the threads to terminate. */ > { > int i; > for (i = 0; i < THREAD_COUNT; i++) > assert (pthread_join (threads[i], NULL) == 0); > } > > /* Finally, determine whether the threads produced the same sequence of > random() results. */ > { > int expected_index; > int result_index[THREAD_COUNT]; > int i; > > for (i = 0; i < THREAD_COUNT; i++) > result_index[i] = 0; > > for (expected_index = 0; expected_index < REPEAT_COUNT; expected_index++) > { > long expected_value = expected[expected_index]; > > for (i = 0; i < THREAD_COUNT; i++) > { > if (thread_results[i][result_index[i]] == expected_value) > { > result_index[i]++; > break; > } > } > if (i == THREAD_COUNT) > { > if (expected_index == 0) > { > /* This occurs on platforms like OpenBSD, where srandom() has no > effect and random() always return non-deterministic values. > Mark the test as SKIP. */ > fprintf (stderr, "Skipping test: random() is non-deterministic.\n"); > return 77; > } > else > { > fprintf (stderr, "Expected value #%d not found in multithreaded results.\n", > expected_index); > return 1; > } > } > } > } > > return 0; > } > > -- > Problem reports: https://cygwin.com/problems.html > FAQ: https://cygwin.com/faq/ > Documentation: https://cygwin.com/docs.html > Unsubscribe info: https://cygwin.com/ml/#unsubscribe-simple -- Problem reports: https://cygwin.com/problems.html FAQ: https://cygwin.com/faq/ Documentation: https://cygwin.com/docs.html Unsubscribe info: https://cygwin.com/ml/#unsubscribe-simple