delorie.com/archives/browse.cgi   search  
Mail Archives: cygwin/2004/04/07/00:46:49

Mailing-List: contact cygwin-help AT cygwin DOT com; run by ezmlm
List-Subscribe: <mailto:cygwin-subscribe AT cygwin DOT com>
List-Archive: <http://sources.redhat.com/ml/cygwin/>
List-Post: <mailto:cygwin AT cygwin DOT com>
List-Help: <mailto:cygwin-help AT cygwin DOT com>, <http://sources.redhat.com/ml/#faqs>
Sender: cygwin-owner AT cygwin DOT com
Mail-Followup-To: cygwin AT cygwin DOT com
Delivered-To: mailing list cygwin AT cygwin DOT com
Date: Tue, 6 Apr 2004 23:43:53 -0500
From: Oleg Ostrozhansky <oleg AT olegos DOT com>
To: cygwin AT cygwin DOT com
Subject: reentrant functions
Message-ID: <20040407044353.GA6805@panix.com>
Mime-Version: 1.0
User-Agent: Mutt/1.4.2.1i

I have a question about writing a multi-threaded program (using POSIX.1
threads) in Cygwin.  "info libc" has a nice chapter about reentrancy,
which talks about _<func>_r reentrant variants for functions that are
not thread-safe.  But when as an example I try using _gets_r(), I'm
getting a link error that this function does not exist:

~ $ gcc -g threadtest.c
/cygdrive/c/.../cc3s8dTu.o(.text+0xb4): In function `main':
threadtest.c:26: undefined reference to `__gets_r'
collect2: ld returned 1 exit status
~ $

I can't find it anywhere in the libraries.  The prototype is in stdio.h,
so the compile step works.  What do I need to do to make it work?

I've just updated to the latest Cygwin (1.5.9) and gcc (3.3.1), with no
change (running under Windows 2000 and Windows XP).

Here are a couple of other notes that I had while working on this:

- "info libc" mentions the use of struct _reent, and that it needs to be
  initialized for each process (shouldn't it be "thread" here? As well
  as in a few other places).  Unfortunately it neglects to say how to
  initialize it.  I found _REENT_INIT and _REENT_INIT_PTR macros in
  sys/reent.h.  It seems that the first one needs to be used as
      struct _reent thread_one = _REENT_INIT(thread_one);

- I don't understand this part, as a possible way of achieving
  reentrancy (also from "info libc"):

    2. Ensure that each thread of execution control has a pointer
    to its own unique reentrancy structure in the global variable
    `_impure_ptr', and call the standard library subroutines.

  How is it possible for two threads sharing global variables have
  the same global variable point to different structures?  (reentrancy
  is only an issue when address space is shared, right? so it isn't
  possible to have separate _impure_ptr's, as I understand.  What am
  I understanding wrong?)

- Is it ok to use the regular functions, that use the global reentrancy
  structure, in one thread (or at least until multiple threads are
  created), and _<func>_r ones in the other ones?

- In stdio.h there is a typo where the comment says "Recursive versions of
  the above."  I believe it should say "Reentrant", not "Recursive".


Thank you!

  Oleg


--
Unsubscribe info:      http://cygwin.com/ml/#unsubscribe-simple
Problem reports:       http://cygwin.com/problems.html
Documentation:         http://cygwin.com/docs.html
FAQ:                   http://cygwin.com/faq/

- Raw text -


  webmaster     delorie software   privacy  
  Copyright © 2019   by DJ Delorie     Updated Jul 2019