00001 /******************************************************************************* 00002 * Copyright (C) 2005 Vintela, Inc. All rights reserved. 00003 * 00004 * Redistribution and use in source and binary forms, with or without 00005 * modification, are permitted provided that the following conditions are met: 00006 * 00007 * - Redistributions of source code must retain the above copyright notice, 00008 * this list of conditions and the following disclaimer. 00009 * 00010 * - Redistributions in binary form must reproduce the above copyright notice, 00011 * this list of conditions and the following disclaimer in the documentation 00012 * and/or other materials provided with the distribution. 00013 * 00014 * - Neither the name of Vintela, Inc. nor the names of its 00015 * contributors may be used to endorse or promote products derived from this 00016 * software without specific prior written permission. 00017 * 00018 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ``AS IS'' 00019 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 00020 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 00021 * ARE DISCLAIMED. IN NO EVENT SHALL Vintela, Inc. OR THE CONTRIBUTORS 00022 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 00023 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 00024 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 00025 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 00026 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 00027 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 00028 * POSSIBILITY OF SUCH DAMAGE. 00029 *******************************************************************************/ 00030 00035 #include "OW_config.h" 00036 #include "OW_ThreadOnce.hpp" 00037 #include "OW_Assertion.hpp" 00038 00039 #if defined(OW_WIN32) 00040 00041 #include <stdio.h> 00042 00043 namespace OW_NAMESPACE 00044 { 00045 00046 void callOnce(OnceFlag& flag, void (*func)()) 00047 { 00048 // this is the double-checked locking pattern, but with a bit more strength than normally implemented :-) 00049 if (InterlockedCompareExchange(&flag, 1, 1) == 0) 00050 { 00051 wchar_t mutexName[MAX_PATH]; 00052 _snwprintf(mutexName, MAX_PATH, L"%X-%p-587ccea9-c95a-4e81-ac51-ab0ddc6cef63", GetCurrentProcessId(), &flag); 00053 mutexName[MAX_PATH - 1] = 0; 00054 00055 HANDLE mutex = CreateMutexW(NULL, FALSE, mutexName); 00056 OW_ASSERT(mutex != NULL); 00057 00058 int res = 0; 00059 res = WaitForSingleObject(mutex, INFINITE); 00060 OW_ASSERT(res == WAIT_OBJECT_0); 00061 00062 if (InterlockedCompareExchange(&flag, 1, 1) == 0) 00063 { 00064 try 00065 { 00066 func(); 00067 } 00068 catch (...) 00069 { 00070 res = ReleaseMutex(mutex); 00071 OW_ASSERT(res); 00072 res = CloseHandle(mutex); 00073 OW_ASSERT(res); 00074 throw; 00075 } 00076 InterlockedExchange(&flag, 1); 00077 } 00078 00079 res = ReleaseMutex(mutex); 00080 OW_ASSERT(res); 00081 res = CloseHandle(mutex); 00082 OW_ASSERT(res); 00083 } 00084 } 00085 00086 00087 } // end namespace OW_NAMESPACE 00088 00089 #endif 00090 00091