1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64
|
// Copyright 2019 The Emscripten Authors. All rights reserved.
// Emscripten is available under two separate licenses, the MIT license and the
// University of Illinois/NCSA Open Source License. Both these licenses can be
// found in the LICENSE file.
#include <stdatomic.h>
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <assert.h>
#include <emscripten.h>
#define NUM_THREADS 2
#define TOTAL 100
static _Atomic int sum;
void *ThreadMain(void *arg) {
for (int i = 0; i < TOTAL; i++) {
// wait for a change, so we see interleaved processing.
int last = ++sum;
while (sum == last) {}
}
pthread_exit((void*)TOTAL);
}
pthread_t thread[NUM_THREADS];
void CreateThread(long i) {
int rc = pthread_create(&thread[i], NULL, ThreadMain, (void*)i);
assert(rc == 0);
}
void main_iter() {
static int main_adds = 0;
int worker_adds = sum++ - main_adds++;
printf("main iter %d : %d\n", main_adds, worker_adds);
if (worker_adds == NUM_THREADS * TOTAL) {
printf("done!\n");
#ifndef ALLOW_SYNC
emscripten_cancel_main_loop();
#endif
exit(0);
}
}
int main() {
// Create initial threads.
for (long i = 0; i < NUM_THREADS; ++i) {
CreateThread(i);
}
// if we don't allow sync pthread creation, the event loop must be reached for
// the worker to start up.
#ifndef ALLOW_SYNC
emscripten_set_main_loop(main_iter, 0, 0);
#else
while (1) {
main_iter();
}
#endif
return 0;
}
|