/* Copyright (c) 2001, Stanford University * All rights reserved. * * See the file LICENSE.txt for information on redistributing this software. */ #include "cr_spu.h" #include "chromium.h" #include "cr_error.h" #include "cr_net.h" #include "cr_rand.h" #include "server_dispatch.h" #include "server.h" #include "cr_mem.h" #include "cr_string.h" GLint SERVER_DISPATCH_APIENTRY crServerDispatchCreateContext(const char *dpyName, GLint visualBits, GLint shareCtx) { return crServerDispatchCreateContextEx(dpyName, visualBits, shareCtx, -1, -1); } GLint crServerDispatchCreateContextEx(const char *dpyName, GLint visualBits, GLint shareCtx, GLint preloadCtxID, int32_t internalID) { GLint retVal = -1; CRContext *newCtx; CRCreateInfo_t *pCreateInfo; if (shareCtx > 0) { crWarning("CRServer: context sharing not implemented."); shareCtx = 0; } /* Since the Cr server serialized all incoming clients/contexts into * one outgoing GL stream, we only need to create one context for the * head SPU. We'll only have to make it current once too, below. */ if (cr_server.firstCallCreateContext) { cr_server.SpuContextVisBits = visualBits; cr_server.SpuContext = cr_server.head_spu->dispatch_table. CreateContext(dpyName, cr_server.SpuContextVisBits, shareCtx); if (cr_server.SpuContext < 0) { crWarning("crServerDispatchCreateContext() failed."); return -1; } cr_server.firstCallCreateContext = GL_FALSE; } else { /* second or third or ... context */ if ((visualBits & cr_server.SpuContextVisBits) != visualBits) { int oldSpuContext; /* the new context needs new visual attributes */ cr_server.SpuContextVisBits |= visualBits; crDebug("crServerDispatchCreateContext requires new visual (0x%x).", cr_server.SpuContextVisBits); /* Here, we used to just destroy the old rendering context. * Unfortunately, this had the side effect of destroying * all display lists and textures that had been loaded on * the old context as well. * * Now, first try to create a new context, with a suitable * visual, sharing display lists and textures with the * old context. Then destroy the old context. */ /* create new rendering context with suitable visual */ oldSpuContext = cr_server.SpuContext; cr_server.SpuContext = cr_server.head_spu->dispatch_table. CreateContext(dpyName, cr_server.SpuContextVisBits, cr_server.SpuContext); /* destroy old rendering context */ cr_server.head_spu->dispatch_table.DestroyContext(oldSpuContext); if (cr_server.SpuContext < 0) { crWarning("crServerDispatchCreateContext() failed."); return -1; } } } /* Now create a new state-tracker context and initialize the * dispatch function pointers. */ newCtx = crStateCreateContextEx(&cr_server.limits, visualBits, NULL, internalID); if (newCtx) { crStateSetCurrentPointers( newCtx, &(cr_server.current) ); crStateResetCurrentPointers(&(cr_server.current)); retVal = preloadCtxID<0 ? crServerGenerateID(&cr_server.idsPool.freeContextID) : preloadCtxID; crHashtableAdd(cr_server.contextTable, retVal, newCtx); pCreateInfo = (CRCreateInfo_t *) crAlloc(sizeof(CRCreateInfo_t)); pCreateInfo->pszDpyName = dpyName ? crStrdup(dpyName) : NULL; pCreateInfo->visualBits = visualBits; pCreateInfo->internalID = newCtx->id; crHashtableAdd(cr_server.pContextCreateInfoTable, retVal, pCreateInfo); } if (retVal != -1 && !cr_server.bIsInLoadingState) { int pos; for (pos = 0; pos < CR_MAX_CONTEXTS; pos++) { if (cr_server.curClient->contextList[pos] == 0) { cr_server.curClient->contextList[pos] = retVal; break; } } } crServerReturnValue( &retVal, sizeof(retVal) ); return retVal; } static int crServerRemoveClientContext(CRClient *pClient, GLint ctx) { int pos; for (pos = 0; pos < CR_MAX_CONTEXTS; ++pos) { if (pClient->contextList[pos] == ctx) { pClient->contextList[pos] = 0; return true; } } return false; } void SERVER_DISPATCH_APIENTRY crServerDispatchDestroyContext( GLint ctx ) { CRContext *crCtx; int32_t client; CRClientNode *pNode; int found=false; crCtx = (CRContext *) crHashtableSearch(cr_server.contextTable, ctx); if (!crCtx) { crWarning("CRServer: DestroyContext invalid context %d", ctx); return; } crDebug("CRServer: DestroyContext context %d", ctx); crHashtableDelete(cr_server.contextTable, ctx, NULL); crStateDestroyContext( crCtx ); crHashtableDelete(cr_server.pContextCreateInfoTable, ctx, crServerCreateInfoDeleteCB); if (cr_server.curClient) { /* If we delete our current context, default back to the null context */ if (cr_server.curClient->currentCtx == crCtx) { cr_server.curClient->currentContextNumber = -1; cr_server.curClient->currentCtx = cr_server.DummyContext; } found = crServerRemoveClientContext(cr_server.curClient, ctx); /*Some application call destroy context not in a thread where it was created...have do deal with it.*/ if (!found) { for (client=0; clientpClient, ctx); pNode = pNode->next; } } CRASSERT(found); } /*Make sure this context isn't active in other clients*/ for (client=0; clientcurrentCtx == crCtx) { cr_server.clients[client]->currentContextNumber = -1; cr_server.clients[client]->currentCtx = cr_server.DummyContext; } } pNode=cr_server.pCleanupClient; while (pNode) { if (pNode->pClient->currentCtx == crCtx) { pNode->pClient->currentContextNumber = -1; pNode->pClient->currentCtx = cr_server.DummyContext; } pNode = pNode->next; } } void SERVER_DISPATCH_APIENTRY crServerDispatchMakeCurrent( GLint window, GLint nativeWindow, GLint context ) { CRMuralInfo *mural, *oldMural; CRContext *ctx; if (context >= 0 && window >= 0) { mural = (CRMuralInfo *) crHashtableSearch(cr_server.muralTable, window); if (!mural) { crWarning("CRServer: invalid window %d passed to crServerDispatchMakeCurrent()", window); return; } /* Update the state tracker's current context */ ctx = (CRContext *) crHashtableSearch(cr_server.contextTable, context); if (!ctx) { crWarning("CRserver: NULL context in MakeCurrent %d", context); return; } } else { oldMural = (CRMuralInfo *) crHashtableSearch(cr_server.muralTable, cr_server.currentWindow); if (oldMural && oldMural->bUseFBO && crServerSupportRedirMuralFBO()) { if (!crStateGetCurrent()->framebufferobject.drawFB) { cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_DRAW_FRAMEBUFFER, 0); } if (!crStateGetCurrent()->framebufferobject.readFB) { cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_READ_FRAMEBUFFER, 0); } } ctx = cr_server.DummyContext; window = -1; mural = NULL; return; } /* crDebug("**** %s client %d curCtx=%d curWin=%d", __func__, cr_server.curClient->number, ctxPos, window); */ cr_server.curClient->currentContextNumber = context; cr_server.curClient->currentCtx = ctx; cr_server.curClient->currentMural = mural; cr_server.curClient->currentWindow = window; CRASSERT(cr_server.curClient->currentCtx); /* This is a hack to force updating the 'current' attribs */ crStateUpdateColorBits(); if (ctx) crStateSetCurrentPointers( ctx, &(cr_server.current) ); /* check if being made current for first time, update viewport */ #if 0 if (ctx) { /* initialize the viewport */ if (ctx->viewport.viewportW == 0) { ctx->viewport.viewportW = mural->width; ctx->viewport.viewportH = mural->height; ctx->viewport.scissorW = mural->width; ctx->viewport.scissorH = mural->height; } } #endif /* crDebug("**** %s currentWindow %d newWindow %d", __func__, cr_server.currentWindow, window); */ oldMural = (CRMuralInfo *) crHashtableSearch(cr_server.muralTable, cr_server.currentWindow); if (1/*cr_server.firstCallMakeCurrent || cr_server.currentWindow != window || cr_server.currentNativeWindow != nativeWindow*/) { /* Since the cr server serialized all incoming contexts/clients into * one output stream of GL commands, we only need to call the head * SPU's MakeCurrent() function once. * BUT, if we're rendering to multiple windows, we do have to issue * MakeCurrent() calls sometimes. The same GL context will always be * used though. */ cr_server.head_spu->dispatch_table.MakeCurrent( mural->spuWindow, nativeWindow, cr_server.SpuContext ); cr_server.firstCallMakeCurrent = GL_FALSE; cr_server.currentWindow = window; cr_server.currentNativeWindow = nativeWindow; } /* This used to be earlier, after crStateUpdateColorBits() call */ crStateMakeCurrent( ctx ); if (oldMural != mural && crServerSupportRedirMuralFBO()) { if (!crStateGetCurrent()->framebufferobject.drawFB) { cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_DRAW_FRAMEBUFFER, mural->bUseFBO ? mural->idFBO:0); } if (!crStateGetCurrent()->framebufferobject.readFB) { cr_server.head_spu->dispatch_table.BindFramebufferEXT(GL_READ_FRAMEBUFFER, mural->bUseFBO ? mural->idFBO:0); } } if (!mural->bUseFBO) { ctx->buffer.width = mural->width; ctx->buffer.height = mural->height; } else { ctx->buffer.width = 0; ctx->buffer.height = 0; } }