blob: f742d0521e1c35f26d89c6b6890d5a400c41d69a [file] [log] [blame]
Guido van Rossum34679b71993-01-26 13:33:44 +00001
Guido van Rossum1d5735e1994-08-30 08:27:36 +00002/* Thread package.
3 This is intended to be usable independently from Python.
4 The implementation for system foobar is in a file thread_foobar.h
5 which is included by this file dependent on config settings.
6 Stuff shared by all thread_*.h files is collected here. */
7
Martin v. Löwiscdc44512002-01-12 11:05:12 +00008#include "Python.h"
Eric Snow2ebc5ce2017-09-07 23:51:28 -06009#include "internal/pystate.h"
Guido van Rossum1d5735e1994-08-30 08:27:36 +000010
Matthias Klosea2542be2004-08-16 11:35:51 +000011#ifndef _POSIX_THREADS
12/* This means pthreads are not implemented in libc headers, hence the macro
13 not present in unistd.h. But they still can be implemented as an external
14 library (e.g. gnu pth in pthread emulation) */
15# ifdef HAVE_PTHREAD_H
16# include <pthread.h> /* _POSIX_THREADS */
17# endif
18#endif
19
Guido van Rossum2571cc81999-04-07 16:07:23 +000020#ifndef DONT_HAVE_STDIO_H
Guido van Rossum1d5735e1994-08-30 08:27:36 +000021#include <stdio.h>
Guido van Rossum2571cc81999-04-07 16:07:23 +000022#endif
Guido van Rossum1d5735e1994-08-30 08:27:36 +000023
Guido van Rossum1d5735e1994-08-30 08:27:36 +000024#include <stdlib.h>
Guido van Rossum1d5735e1994-08-30 08:27:36 +000025
Guido van Rossum49b56061998-10-01 20:42:43 +000026#include "pythread.h"
Guido van Rossum1984f1e1992-08-04 12:41:02 +000027
Guido van Rossum1d5735e1994-08-30 08:27:36 +000028#ifndef _POSIX_THREADS
29
Guido van Rossum539c6622005-09-14 17:49:54 +000030/* Check if we're running on HP-UX and _SC_THREADS is defined. If so, then
Ezio Melotti13925002011-03-16 11:05:33 +020031 enough of the Posix threads package is implemented to support python
Guido van Rossum539c6622005-09-14 17:49:54 +000032 threads.
33
34 This is valid for HP-UX 11.23 running on an ia64 system. If needed, add
Martin Panter7462b6492015-11-02 03:37:02 +000035 a check of __ia64 to verify that we're running on an ia64 system instead
Guido van Rossum539c6622005-09-14 17:49:54 +000036 of a pa-risc system.
37*/
38#ifdef __hpux
39#ifdef _SC_THREADS
40#define _POSIX_THREADS
41#endif
42#endif
43
Sjoerd Mullender66bca321993-12-03 16:54:45 +000044#endif /* _POSIX_THREADS */
Guido van Rossum1984f1e1992-08-04 12:41:02 +000045
Guido van Rossum1984f1e1992-08-04 12:41:02 +000046
Guido van Rossum408027e1996-12-30 16:17:54 +000047#ifdef Py_DEBUG
Guido van Rossum1d5735e1994-08-30 08:27:36 +000048static int thread_debug = 0;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000049#define dprintf(args) (void)((thread_debug & 1) && printf args)
50#define d2printf(args) ((thread_debug & 8) && printf args)
Guido van Rossum1d5735e1994-08-30 08:27:36 +000051#else
52#define dprintf(args)
53#define d2printf(args)
54#endif
55
Guido van Rossum1984f1e1992-08-04 12:41:02 +000056static int initialized;
57
Thomas Wouters8ec68fd2000-07-24 14:39:50 +000058static void PyThread__init_thread(void); /* Forward */
Sjoerd Mullenderaee8bc11992-09-02 11:25:37 +000059
Thomas Wouters73e5a5b2006-06-08 15:35:45 +000060void
61PyThread_init_thread(void)
Sjoerd Mullenderaee8bc11992-09-02 11:25:37 +000062{
Guido van Rossum408027e1996-12-30 16:17:54 +000063#ifdef Py_DEBUG
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000064 char *p = Py_GETENV("PYTHONTHREADDEBUG");
Sjoerd Mullender66bca321993-12-03 16:54:45 +000065
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000066 if (p) {
67 if (*p)
68 thread_debug = atoi(p);
69 else
70 thread_debug = 1;
71 }
Guido van Rossum408027e1996-12-30 16:17:54 +000072#endif /* Py_DEBUG */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000073 if (initialized)
74 return;
75 initialized = 1;
76 dprintf(("PyThread_init_thread called\n"));
77 PyThread__init_thread();
Sjoerd Mullenderaee8bc11992-09-02 11:25:37 +000078}
79
Masayuki Yamamotoaa0aa042017-07-03 20:34:38 +090080#if defined(_POSIX_THREADS)
81# define PYTHREAD_NAME "pthread"
82# include "thread_pthread.h"
83#elif defined(NT_THREADS)
84# define PYTHREAD_NAME "nt"
85# include "thread_nt.h"
86#else
87# error "Require native thread feature. See https://bugs.python.org/issue30832"
Guido van Rossumc3f82b61995-01-17 16:29:31 +000088#endif
89
Guido van Rossum8e9ebfd1997-11-22 21:53:48 +000090
Thomas Wouters0e3f5912006-08-11 14:57:12 +000091/* return the current thread stack size */
92size_t
93PyThread_get_stacksize(void)
94{
Eric Snow2ebc5ce2017-09-07 23:51:28 -060095 return PyThreadState_GET()->interp->pythread_stacksize;
Thomas Wouters0e3f5912006-08-11 14:57:12 +000096}
97
98/* Only platforms defining a THREAD_SET_STACKSIZE() macro
99 in thread_<platform>.h support changing the stack size.
100 Return 0 if stack size is valid,
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000101 -1 if stack size value is invalid,
102 -2 if setting stack size is not supported. */
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000103int
104PyThread_set_stacksize(size_t size)
105{
106#if defined(THREAD_SET_STACKSIZE)
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000107 return THREAD_SET_STACKSIZE(size);
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000108#else
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000109 return -2;
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000110#endif
111}
112
Mark Hammond8d98d2c2003-04-19 15:41:53 +0000113
Tim Petersfda787f2004-10-09 22:33:09 +0000114/* ------------------------------------------------------------------------
115Per-thread data ("key") support.
Mark Hammond8d98d2c2003-04-19 15:41:53 +0000116
Tim Petersfda787f2004-10-09 22:33:09 +0000117Use PyThread_create_key() to create a new key. This is typically shared
118across threads.
119
120Use PyThread_set_key_value(thekey, value) to associate void* value with
121thekey in the current thread. Each thread has a distinct mapping of thekey
122to a void* value. Caution: if the current thread already has a mapping
123for thekey, value is ignored.
124
125Use PyThread_get_key_value(thekey) to retrieve the void* value associated
126with thekey in the current thread. This returns NULL if no value is
127associated with thekey in the current thread.
128
129Use PyThread_delete_key_value(thekey) to forget the current thread's associated
130value for thekey. PyThread_delete_key(thekey) forgets the values associated
131with thekey across *all* threads.
132
133While some of these functions have error-return values, none set any
134Python exception.
135
136None of the functions does memory management on behalf of the void* values.
137You need to allocate and deallocate them yourself. If the void* values
138happen to be PyObject*, these functions don't do refcount operations on
139them either.
140
141The GIL does not need to be held when calling these functions; they supply
142their own locking. This isn't true of PyThread_create_key(), though (see
143next paragraph).
144
145There's a hidden assumption that PyThread_create_key() will be called before
146any of the other functions are called. There's also a hidden assumption
147that calls to PyThread_create_key() are serialized externally.
148------------------------------------------------------------------------ */
149
Victor Stinner754851f2011-04-19 23:58:51 +0200150
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200151PyDoc_STRVAR(threadinfo__doc__,
152"sys.thread_info\n\
153\n\
154A struct sequence holding information about the thread implementation.");
155
156static PyStructSequence_Field threadinfo_fields[] = {
157 {"name", "name of the thread implementation"},
158 {"lock", "name of the lock implementation"},
159 {"version", "name and version of the thread library"},
160 {0}
161};
162
163static PyStructSequence_Desc threadinfo_desc = {
164 "sys.thread_info", /* name */
165 threadinfo__doc__, /* doc */
166 threadinfo_fields, /* fields */
167 3
168};
169
170static PyTypeObject ThreadInfoType;
171
Victor Stinner754851f2011-04-19 23:58:51 +0200172PyObject*
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200173PyThread_GetInfo(void)
Victor Stinner754851f2011-04-19 23:58:51 +0200174{
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200175 PyObject *threadinfo, *value;
176 int pos = 0;
Victor Stinnere07f5222011-04-20 12:23:26 +0200177#if (defined(_POSIX_THREADS) && defined(HAVE_CONFSTR) \
178 && defined(_CS_GNU_LIBPTHREAD_VERSION))
Victor Stinner754851f2011-04-19 23:58:51 +0200179 char buffer[255];
180 int len;
Victor Stinnere07f5222011-04-20 12:23:26 +0200181#endif
Victor Stinner754851f2011-04-19 23:58:51 +0200182
Victor Stinner1c8f0592013-07-22 22:24:54 +0200183 if (ThreadInfoType.tp_name == 0) {
184 if (PyStructSequence_InitType2(&ThreadInfoType, &threadinfo_desc) < 0)
185 return NULL;
186 }
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200187
188 threadinfo = PyStructSequence_New(&ThreadInfoType);
189 if (threadinfo == NULL)
Victor Stinner754851f2011-04-19 23:58:51 +0200190 return NULL;
191
192 value = PyUnicode_FromString(PYTHREAD_NAME);
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200193 if (value == NULL) {
194 Py_DECREF(threadinfo);
195 return NULL;
196 }
197 PyStructSequence_SET_ITEM(threadinfo, pos++, value);
Victor Stinner754851f2011-04-19 23:58:51 +0200198
199#ifdef _POSIX_THREADS
200#ifdef USE_SEMAPHORES
201 value = PyUnicode_FromString("semaphore");
202#else
203 value = PyUnicode_FromString("mutex+cond");
204#endif
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200205 if (value == NULL) {
206 Py_DECREF(threadinfo);
Victor Stinner754851f2011-04-19 23:58:51 +0200207 return NULL;
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200208 }
209#else
210 Py_INCREF(Py_None);
211 value = Py_None;
212#endif
213 PyStructSequence_SET_ITEM(threadinfo, pos++, value);
Victor Stinner754851f2011-04-19 23:58:51 +0200214
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200215#if (defined(_POSIX_THREADS) && defined(HAVE_CONFSTR) \
216 && defined(_CS_GNU_LIBPTHREAD_VERSION))
217 value = NULL;
Victor Stinner754851f2011-04-19 23:58:51 +0200218 len = confstr(_CS_GNU_LIBPTHREAD_VERSION, buffer, sizeof(buffer));
Victor Stinner98ea54c2014-08-15 23:30:40 +0200219 if (1 < len && (size_t)len < sizeof(buffer)) {
Victor Stinner754851f2011-04-19 23:58:51 +0200220 value = PyUnicode_DecodeFSDefaultAndSize(buffer, len-1);
221 if (value == NULL)
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200222 PyErr_Clear();
Victor Stinner754851f2011-04-19 23:58:51 +0200223 }
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200224 if (value == NULL)
Victor Stinner754851f2011-04-19 23:58:51 +0200225#endif
Victor Stinnerd5c355c2011-04-30 14:53:09 +0200226 {
227 Py_INCREF(Py_None);
228 value = Py_None;
229 }
230 PyStructSequence_SET_ITEM(threadinfo, pos++, value);
231 return threadinfo;
Victor Stinner754851f2011-04-19 23:58:51 +0200232}