* ntdll.h (struct _SEMAPHORE_BASIC_INFORMATION): Define.
(enum _SEMAPHORE_INFORMATION_CLASS): Define. (NtQuerySemaphore): Declare. * thread.h (class semaphore): Add member startvalue. (semaphore::fixup_before_fork): New inline method. (semaphore::_fixup_before_fork): Declare. * thread.cc (MTinterface::fixup_before_fork): Additionally call semaphore::fixup_before_fork. (semaphore::semaphore): Set currentvalue to -1. Set startvalue to incoming initializer value. (semaphore::_getvalue): Just query semaphore using NtQuerySemaphore rather then using WFSO/Release. (semaphore::_post): Drop setting currentvalue. It's not thread-safe. (semaphore::_trywait): Ditto. (semaphore::_timedwait): Ditto. (semaphore::_wait): Ditto. (semaphore::_fixup_before_fork): New method, setting currentvalue from actual windows semaphore right before fork. (semaphore::_fixup_after_fork): Drop kludge from 2013-07-10. Drop FIXME comment.
This commit is contained in:
		| @@ -1,3 +1,26 @@ | |||||||
|  | 2013-07-23  Corinna Vinschen  <corinna@vinschen.de> | ||||||
|  |  | ||||||
|  | 	* ntdll.h (struct _SEMAPHORE_BASIC_INFORMATION): Define. | ||||||
|  | 	(enum _SEMAPHORE_INFORMATION_CLASS): Define. | ||||||
|  | 	(NtQuerySemaphore): Declare. | ||||||
|  | 	* thread.h (class semaphore): Add member startvalue. | ||||||
|  | 	(semaphore::fixup_before_fork): New inline method. | ||||||
|  | 	(semaphore::_fixup_before_fork): Declare. | ||||||
|  | 	* thread.cc (MTinterface::fixup_before_fork): Additionally call | ||||||
|  | 	semaphore::fixup_before_fork. | ||||||
|  | 	(semaphore::semaphore): Set currentvalue to -1.  Set startvalue to | ||||||
|  | 	incoming initializer value. | ||||||
|  | 	(semaphore::_getvalue): Just query semaphore using NtQuerySemaphore | ||||||
|  | 	rather then using WFSO/Release. | ||||||
|  | 	(semaphore::_post): Drop setting currentvalue.  It's not thread-safe. | ||||||
|  | 	(semaphore::_trywait): Ditto. | ||||||
|  | 	(semaphore::_timedwait): Ditto. | ||||||
|  | 	(semaphore::_wait): Ditto. | ||||||
|  | 	(semaphore::_fixup_before_fork): New method, setting currentvalue from | ||||||
|  | 	actual windows semaphore right before fork. | ||||||
|  | 	(semaphore::_fixup_after_fork): Drop kludge from 2013-07-10.  Drop | ||||||
|  | 	FIXME comment. | ||||||
|  |  | ||||||
| 2013-07-23  Corinna Vinschen  <corinna@vinschen.de> | 2013-07-23  Corinna Vinschen  <corinna@vinschen.de> | ||||||
|  |  | ||||||
| 	* cygtls.cc (well_known_dlls): Add kernelbase.dll. | 	* cygtls.cc (well_known_dlls): Add kernelbase.dll. | ||||||
|   | |||||||
| @@ -1102,6 +1102,18 @@ typedef enum _EVENT_INFORMATION_CLASS | |||||||
|   EventBasicInformation = 0 |   EventBasicInformation = 0 | ||||||
| } EVENT_INFORMATION_CLASS, *PEVENT_INFORMATION_CLASS; | } EVENT_INFORMATION_CLASS, *PEVENT_INFORMATION_CLASS; | ||||||
|  |  | ||||||
|  | /* Checked on 64 bit. */ | ||||||
|  | typedef struct _SEMAPHORE_BASIC_INFORMATION | ||||||
|  | { | ||||||
|  |   LONG CurrentCount; | ||||||
|  |   LONG MaximumCount; | ||||||
|  | } SEMAPHORE_BASIC_INFORMATION, *PSEMAPHORE_BASIC_INFORMATION; | ||||||
|  |  | ||||||
|  | typedef enum _SEMAPHORE_INFORMATION_CLASS | ||||||
|  | { | ||||||
|  |   SemaphoreBasicInformation = 0 | ||||||
|  | } SEMAPHORE_INFORMATION_CLASS, *PSEMAPHORE_INFORMATION_CLASS; | ||||||
|  |  | ||||||
| typedef enum _THREAD_INFORMATION_CLASS | typedef enum _THREAD_INFORMATION_CLASS | ||||||
| { | { | ||||||
|   ThreadBasicInformation = 0, |   ThreadBasicInformation = 0, | ||||||
| @@ -1275,6 +1287,8 @@ extern "C" | |||||||
| 					  PVOID, ULONG, PULONG); | 					  PVOID, ULONG, PULONG); | ||||||
|   NTSTATUS NTAPI NtQueryObject (HANDLE, OBJECT_INFORMATION_CLASS, VOID *, |   NTSTATUS NTAPI NtQueryObject (HANDLE, OBJECT_INFORMATION_CLASS, VOID *, | ||||||
| 				ULONG, ULONG *); | 				ULONG, ULONG *); | ||||||
|  |   NTSTATUS NTAPI NtQuerySemaphore (HANDLE, SEMAPHORE_INFORMATION_CLASS, | ||||||
|  | 				   PVOID, ULONG, PULONG); | ||||||
|   NTSTATUS NTAPI NtQuerySystemInformation (SYSTEM_INFORMATION_CLASS, |   NTSTATUS NTAPI NtQuerySystemInformation (SYSTEM_INFORMATION_CLASS, | ||||||
| 					   PVOID, ULONG, PULONG); | 					   PVOID, ULONG, PULONG); | ||||||
|   NTSTATUS WINAPI NtQuerySystemTime (PLARGE_INTEGER); |   NTSTATUS WINAPI NtQuerySystemTime (PLARGE_INTEGER); | ||||||
|   | |||||||
| @@ -316,6 +316,7 @@ void | |||||||
| MTinterface::fixup_before_fork () | MTinterface::fixup_before_fork () | ||||||
| { | { | ||||||
|   pthread_key::fixup_before_fork (); |   pthread_key::fixup_before_fork (); | ||||||
|  |   semaphore::fixup_before_fork (); | ||||||
| } | } | ||||||
|  |  | ||||||
| /* This function is called from a single threaded process */ | /* This function is called from a single threaded process */ | ||||||
| @@ -3376,7 +3377,8 @@ List<semaphore> semaphore::semaphores; | |||||||
| semaphore::semaphore (int pshared, unsigned int value) | semaphore::semaphore (int pshared, unsigned int value) | ||||||
| : verifyable_object (SEM_MAGIC), | : verifyable_object (SEM_MAGIC), | ||||||
|   shared (pshared), |   shared (pshared), | ||||||
|   currentvalue (value), |   currentvalue (-1), | ||||||
|  |   startvalue (value), | ||||||
|   fd (-1), |   fd (-1), | ||||||
|   hash (0ULL), |   hash (0ULL), | ||||||
|   sem (NULL) |   sem (NULL) | ||||||
| @@ -3394,7 +3396,8 @@ semaphore::semaphore (unsigned long long shash, LUID sluid, int sfd, | |||||||
| 		      sem_t *ssem, int oflag, mode_t mode, unsigned int value) | 		      sem_t *ssem, int oflag, mode_t mode, unsigned int value) | ||||||
| : verifyable_object (SEM_MAGIC), | : verifyable_object (SEM_MAGIC), | ||||||
|   shared (PTHREAD_PROCESS_SHARED), |   shared (PTHREAD_PROCESS_SHARED), | ||||||
|   currentvalue (value),		/* Unused for named semaphores. */ |   currentvalue (-1),		/* Unused for named semaphores. */ | ||||||
|  |   startvalue (value), | ||||||
|   fd (sfd), |   fd (sfd), | ||||||
|   hash (shash), |   hash (shash), | ||||||
|   luid (sluid), |   luid (sluid), | ||||||
| @@ -3428,29 +3431,21 @@ semaphore::~semaphore () | |||||||
| void | void | ||||||
| semaphore::_post () | semaphore::_post () | ||||||
| { | { | ||||||
|   if (ReleaseSemaphore (win32_obj_id, 1, ¤tvalue)) |   LONG dummy; | ||||||
|     currentvalue++; |   ReleaseSemaphore (win32_obj_id, 1, &dummy); | ||||||
| } | } | ||||||
|  |  | ||||||
| int | int | ||||||
| semaphore::_getvalue (int *sval) | semaphore::_getvalue (int *sval) | ||||||
| { | { | ||||||
|   LONG val; |   NTSTATUS status; | ||||||
|  |   SEMAPHORE_BASIC_INFORMATION sbi; | ||||||
|  |  | ||||||
|   switch (WaitForSingleObject (win32_obj_id, 0)) |   status = NtQuerySemaphore (win32_obj_id, SemaphoreBasicInformation, &sbi, | ||||||
|     { | 			     sizeof sbi, NULL); | ||||||
|       case WAIT_OBJECT_0: |   if (NT_SUCCESS (status)) | ||||||
| 	ReleaseSemaphore (win32_obj_id, 1, &val); |     return sbi.CurrentCount; | ||||||
| 	*sval = val + 1; |   return startvalue; | ||||||
| 	break; |  | ||||||
|       case WAIT_TIMEOUT: |  | ||||||
| 	*sval = 0; |  | ||||||
| 	break; |  | ||||||
|       default: |  | ||||||
| 	set_errno (EAGAIN); |  | ||||||
| 	return -1; |  | ||||||
|     } |  | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| int | int | ||||||
| @@ -3463,7 +3458,6 @@ semaphore::_trywait () | |||||||
|       set_errno (EAGAIN); |       set_errno (EAGAIN); | ||||||
|       return -1; |       return -1; | ||||||
|     } |     } | ||||||
|   currentvalue--; |  | ||||||
|   return 0; |   return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -3489,7 +3483,6 @@ semaphore::_timedwait (const struct timespec *abstime) | |||||||
|   switch (cygwait (win32_obj_id, &timeout, cw_cancel | cw_cancel_self | cw_sig_eintr)) |   switch (cygwait (win32_obj_id, &timeout, cw_cancel | cw_cancel_self | cw_sig_eintr)) | ||||||
|     { |     { | ||||||
|     case WAIT_OBJECT_0: |     case WAIT_OBJECT_0: | ||||||
|       currentvalue--; |  | ||||||
|       break; |       break; | ||||||
|     case WAIT_SIGNALED: |     case WAIT_SIGNALED: | ||||||
|       set_errno (EINTR); |       set_errno (EINTR); | ||||||
| @@ -3511,7 +3504,6 @@ semaphore::_wait () | |||||||
|   switch (cygwait (win32_obj_id, cw_infinite, cw_cancel | cw_cancel_self | cw_sig_eintr)) |   switch (cygwait (win32_obj_id, cw_infinite, cw_cancel | cw_cancel_self | cw_sig_eintr)) | ||||||
|     { |     { | ||||||
|     case WAIT_OBJECT_0: |     case WAIT_OBJECT_0: | ||||||
|       currentvalue--; |  | ||||||
|       break; |       break; | ||||||
|     case WAIT_SIGNALED: |     case WAIT_SIGNALED: | ||||||
|       set_errno (EINTR); |       set_errno (EINTR); | ||||||
| @@ -3523,19 +3515,31 @@ semaphore::_wait () | |||||||
|   return 0; |   return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | void | ||||||
|  | semaphore::_fixup_before_fork () | ||||||
|  | { | ||||||
|  |   NTSTATUS status; | ||||||
|  |   SEMAPHORE_BASIC_INFORMATION sbi; | ||||||
|  |  | ||||||
|  |   status = NtQuerySemaphore (win32_obj_id, SemaphoreBasicInformation, &sbi, | ||||||
|  | 			     sizeof sbi, NULL); | ||||||
|  |   if (NT_SUCCESS (status)) | ||||||
|  |     currentvalue = sbi.CurrentCount; | ||||||
|  |   else | ||||||
|  |     currentvalue = startvalue; | ||||||
|  | } | ||||||
|  |  | ||||||
| void | void | ||||||
| semaphore::_fixup_after_fork () | semaphore::_fixup_after_fork () | ||||||
| { | { | ||||||
|   if (shared == PTHREAD_PROCESS_PRIVATE) |   if (shared == PTHREAD_PROCESS_PRIVATE) | ||||||
|     { |     { | ||||||
|       pthread_printf ("sem %p", this); |       pthread_printf ("sem %p", this); | ||||||
|       if (!currentvalue) |  | ||||||
| 	currentvalue = 1; |  | ||||||
|       /* FIXME: duplicate code here and in the constructor. */ |  | ||||||
|       win32_obj_id = ::CreateSemaphore (&sec_none_nih, currentvalue, |       win32_obj_id = ::CreateSemaphore (&sec_none_nih, currentvalue, | ||||||
| 					      INT32_MAX, NULL); | 					INT32_MAX, NULL); | ||||||
|       if (!win32_obj_id) |       if (!win32_obj_id) | ||||||
| 	api_fatal ("failed to create new win32 semaphore, currentvalue %ld, %E", currentvalue); | 	api_fatal ("failed to create new win32 semaphore, " | ||||||
|  | 		   "currentvalue %ld, %E", currentvalue); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -638,6 +638,7 @@ public: | |||||||
|   HANDLE win32_obj_id; |   HANDLE win32_obj_id; | ||||||
|   int shared; |   int shared; | ||||||
|   LONG currentvalue; |   LONG currentvalue; | ||||||
|  |   LONG startvalue; | ||||||
|   int fd; |   int fd; | ||||||
|   unsigned long long hash; |   unsigned long long hash; | ||||||
|   LUID luid; |   LUID luid; | ||||||
| @@ -648,6 +649,10 @@ public: | |||||||
|   ~semaphore (); |   ~semaphore (); | ||||||
|  |  | ||||||
|   class semaphore * next; |   class semaphore * next; | ||||||
|  |   static void fixup_before_fork () | ||||||
|  |   { | ||||||
|  |     semaphores.for_each (&semaphore::_fixup_before_fork); | ||||||
|  |   } | ||||||
|   static void fixup_after_fork () |   static void fixup_after_fork () | ||||||
|   { |   { | ||||||
|     semaphores.fixup_after_fork (); |     semaphores.fixup_after_fork (); | ||||||
| @@ -666,6 +671,7 @@ private: | |||||||
|   int _trywait (); |   int _trywait (); | ||||||
|   int _timedwait (const struct timespec *abstime); |   int _timedwait (const struct timespec *abstime); | ||||||
|  |  | ||||||
|  |   void _fixup_before_fork (); | ||||||
|   void _fixup_after_fork (); |   void _fixup_after_fork (); | ||||||
|   void _terminate (); |   void _terminate (); | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user