blob: c3238537871e7820e2f3b20084b8b2d9beef40e9 [file] [log] [blame]
// Copyright 2016 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 <stdio.h>
#include <string.h>
#include <pthread.h>
#include <GLES2/gl2.h>
#include <math.h>
#include <assert.h>
#include <emscripten/emscripten.h>
#include <emscripten/html5.h>
#include <emscripten/threading.h>
#include <bits/errno.h>
#include <stdlib.h>
pthread_t thread;
EMSCRIPTEN_WEBGL_CONTEXT_HANDLE ctx;
int numThreadsCreated = 0;
int result = 0;
void *ThreadMain(void *arg)
{
printf("ThreadMain\n");
switch(numThreadsCreated)
{
case 1: printf("Thread 1 started: you should see the WebGL canvas fade from black to red.\n"); break;
case 2: printf("Thread 2 started: you should see the WebGL canvas fade from black to green.\n"); break;
case 3: printf("Thread 3 started: you should see the WebGL canvas fade from black to blue.\n"); break;
}
EmscriptenWebGLContextAttributes attr;
emscripten_webgl_init_context_attributes(&attr);
attr.explicitSwapControl = EM_TRUE;
ctx = emscripten_webgl_create_context("#canvas", &attr);
emscripten_webgl_make_context_current(ctx);
double color = 0;
for(int i = 0; i < 100; ++i)
{
color += 0.01;
switch(numThreadsCreated)
{
case 1: glClearColor(color, 0, 0, 1); break;
case 2: glClearColor(0, color, 0, 1); break;
case 3: glClearColor(0, 0, color, 1); break;
}
glClear(GL_COLOR_BUFFER_BIT);
EMSCRIPTEN_RESULT r = emscripten_webgl_commit_frame();
assert(r == EMSCRIPTEN_RESULT_SUCCESS);
double now = emscripten_get_now();
while(emscripten_get_now() - now < 16) /*no-op*/;
}
emscripten_webgl_make_context_current(0);
emscripten_webgl_destroy_context(ctx);
printf("Thread quit\n");
pthread_exit(0);
}
void CreateThread()
{
pthread_attr_t attr;
pthread_attr_init(&attr);
emscripten_pthread_attr_settransferredcanvases(&attr, "#canvas");
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
int rc = pthread_create(&thread, &attr, ThreadMain, 0);
if (rc == ENOSYS)
{
printf("Test Skipped! OffscreenCanvas is not supported!\n");
#ifdef REPORT_RESULT
REPORT_RESULT(1); // But report success, so that runs on non-supporting browsers don't raise noisy errors.
#endif
exit(0);
}
if (rc)
{
printf("Failed to create thread! error: %d\n", rc);
exit(0);
}
pthread_attr_destroy(&attr);
++numThreadsCreated;
}
void *mymain(void*)
{
for(int i = 0; i < 3; ++i)
{
printf("Creating thread %d\n", i+1);
CreateThread();
printf("Waiting for thread to finish.\n");
pthread_join(thread, 0);
thread = 0;
}
printf("All done!\n");
return 0;
}
// Tests that the OffscreenCanvas context can travel from main thread -> thread 1 -> thread 2
// #define TEST_CHAINED_WEBGL_CONTEXT_PASSING
// If set, the OffscreenCanvas is transferred from the main thread directly to thread 2, without giving it to thread 1
// in between.
// #define TRANSFER_TO_CHAINED_THREAD_FROM_MAIN_THREAD
int main()
{
#ifdef TEST_CHAINED_WEBGL_CONTEXT_PASSING
EM_ASM(Module['noExitRuntime'] = true;);
pthread_attr_t attr;
pthread_attr_init(&attr);
#ifndef TRANSFER_TO_CHAINED_THREAD_FROM_MAIN_THREAD
emscripten_pthread_attr_settransferredcanvases(&attr, "#canvas");
#endif
int rc = pthread_create(&thread, &attr, mymain, 0);
if (rc == ENOSYS)
{
printf("Test Skipped! OffscreenCanvas is not supported!\n");
#ifdef REPORT_RESULT
REPORT_RESULT(1); // But report success, so that runs on non-supporting browsers don't raise noisy errors.
#endif
exit(0);
}
#else
mymain(0);
#endif
return 0;
}