| #include "config.h" |
| #include <stdlib.h> |
| #include <stdio.h> |
| |
| #include <libxml/parser.h> |
| #include <libxml/threads.h> |
| |
| #if defined(LIBXML_THREAD_ENABLED) && defined(LIBXML_CATALOG_ENABLED) |
| #include <libxml/catalog.h> |
| #ifdef HAVE_PTHREAD_H |
| #include <pthread.h> |
| #elif defined(_WIN32) |
| #include <windows.h> |
| #endif |
| #include <string.h> |
| #include <assert.h> |
| |
| #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); |
| } |
| |