#include "config.h" #include #include #include #include #if defined(LIBXML_THREAD_ENABLED) && defined(LIBXML_CATALOG_ENABLED) #include #ifdef HAVE_PTHREAD_H #include #elif defined(_WIN32) #include #endif #include #include #define MAX_ARGC 20 #define TEST_REPEAT_COUNT 500 #ifdef HAVE_PTHREAD_H static pthread_t tid[MAX_ARGC]; #elif defined(_WIN32) static HANDLE tid[MAX_ARGC]; #endif typedef struct { const char *filename; int okay; } xmlThreadParams; static const char *catalog = "test/threads/complex.xml"; static xmlThreadParams threadParams[] = { { "test/threads/abc.xml", 0 }, { "test/threads/acb.xml", 0 }, { "test/threads/bac.xml", 0 }, { "test/threads/bca.xml", 0 }, { "test/threads/cab.xml", 0 }, { "test/threads/cba.xml", 0 }, { "test/threads/invalid.xml", 0 } }; static const unsigned int num_threads = sizeof(threadParams) / sizeof(threadParams[0]); static void * thread_specific_data(void *private_data) { xmlDocPtr myDoc; xmlThreadParams *params = (xmlThreadParams *) private_data; const char *filename = params->filename; int okay = 1; int options = 0; if (xmlCheckThreadLocalStorage() != 0) { printf("xmlCheckThreadLocalStorage failed\n"); params->okay = 0; return(NULL); } if (strcmp(filename, "test/threads/invalid.xml") != 0) { options |= XML_PARSE_DTDVALID; } myDoc = xmlReadFile(filename, NULL, options); if (myDoc) { xmlFreeDoc(myDoc); } else { printf("parse failed\n"); okay = 0; } params->okay = okay; return(NULL); } #ifdef _WIN32 static DWORD WINAPI win32_thread_specific_data(void *private_data) { thread_specific_data(private_data); return(0); } #endif #endif /* LIBXML_THREADS_ENABLED */ int main(void) { unsigned int repeat; int status = 0; (void) repeat; xmlInitParser(); if (xmlCheckThreadLocalStorage() != 0) { printf("xmlCheckThreadLocalStorage failed for main thread\n"); return(1); } #if defined(LIBXML_THREAD_ENABLED) && defined(LIBXML_CATALOG_ENABLED) for (repeat = 0;repeat < TEST_REPEAT_COUNT;repeat++) { unsigned int i; int ret; xmlLoadCatalog(catalog); #ifdef HAVE_PTHREAD_H memset(tid, 0xff, sizeof(*tid)*num_threads); for (i = 0; i < num_threads; i++) { ret = pthread_create(&tid[i], NULL, thread_specific_data, (void *) &threadParams[i]); if (ret != 0) { perror("pthread_create"); exit(1); } } for (i = 0; i < num_threads; i++) { void *result; ret = pthread_join(tid[i], &result); if (ret != 0) { perror("pthread_join"); exit(1); } } #elif defined(_WIN32) for (i = 0; i < num_threads; i++) { tid[i] = (HANDLE) -1; } for (i = 0; i < num_threads; i++) { DWORD useless; tid[i] = CreateThread(NULL, 0, win32_thread_specific_data, &threadParams[i], 0, &useless); if (tid[i] == NULL) { perror("CreateThread"); exit(1); } } if (WaitForMultipleObjects (num_threads, tid, TRUE, INFINITE) == WAIT_FAILED) perror ("WaitForMultipleObjects failed"); for (i = 0; i < num_threads; i++) { DWORD exitCode; ret = GetExitCodeThread (tid[i], &exitCode); if (ret == 0) { perror("GetExitCodeThread"); exit(1); } CloseHandle (tid[i]); } #endif /* pthreads */ xmlCatalogCleanup(); for (i = 0; i < num_threads; i++) { if (threadParams[i].okay == 0) { printf("Thread %d handling %s failed\n", i, threadParams[i].filename); status = 1; } } } #endif /* LIBXML_THREADS_ENABLED */ xmlCleanupParser(); return (status); }