rsScriptC.cpp revision 26b2c9f5b0d77b69fb5edf96dab7d57f1de1d594
1326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams/* 2326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * Copyright (C) 2009 The Android Open Source Project 3326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * 4326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * Licensed under the Apache License, Version 2.0 (the "License"); 5326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * you may not use this file except in compliance with the License. 6326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * You may obtain a copy of the License at 7326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * 8326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * http://www.apache.org/licenses/LICENSE-2.0 9326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * 10326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * Unless required by applicable law or agreed to in writing, software 11326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * distributed under the License is distributed on an "AS IS" BASIS, 12326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * See the License for the specific language governing permissions and 14326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * limitations under the License. 15326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams */ 16326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 17326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsContext.h" 18326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsScriptC.h" 19326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsMatrix.h" 209c4e4ca9ff75a7fe18544c83fcf782e46c9b6ac2Joe Onorato#include "utils/Timers.h" 21b26fb04770442244233b630960f419cb154abc77Alex Sakhartchouk#include "utils/StopWatch.h" 22ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei Liaoextern "C" { 23ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei Liao#include "libdex/ZipArchive.h" 24ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei Liao} 251ef8b80bc49f415b93912fe7335f852953bde4a4Jack Palevich 261aa5a4eb81b8b88aeb5d2b6f4c47356fd0a62923Jason Sams#include <GLES/gl.h> 271aa5a4eb81b8b88aeb5d2b6f4c47356fd0a62923Jason Sams#include <GLES/glext.h> 281aa5a4eb81b8b88aeb5d2b6f4c47356fd0a62923Jason Sams 29326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsusing namespace android; 30326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsusing namespace android::renderscript; 31326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 32e57691037aea219562ac686429b4b98202aab7bcJason Sams#define GET_TLS() Context::ScriptTLSStruct * tls = \ 33e57691037aea219562ac686429b4b98202aab7bcJason Sams (Context::ScriptTLSStruct *)pthread_getspecific(Context::gThreadTLSKey); \ 34e57691037aea219562ac686429b4b98202aab7bcJason Sams Context * rsc = tls->mContext; \ 35e57691037aea219562ac686429b4b98202aab7bcJason Sams ScriptC * sc = (ScriptC *) tls->mScript 36e57691037aea219562ac686429b4b98202aab7bcJason Sams 372b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao// Input: cacheDir 382b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao// Input: resName 392b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao// Input: extName 402b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao// 412b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao// Note: cacheFile = resName + extName 422b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao// 432b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao// Output: Returns cachePath == cacheDir + cacheFile 442b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liaochar *genCacheFileName(const char *cacheDir, 452b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao const char *resName, 462b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao const char *extName) { 472b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao char cachePath[512]; 482b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao char cacheFile[sizeof(cachePath)]; 492b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao const size_t kBufLen = sizeof(cachePath) - 1; 502b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao 512b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao cacheFile[0] = '\0'; 522b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // Note: resName today is usually something like 532b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // "/com.android.fountain:raw/fountain" 542b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao if (resName[0] != '/') { 552b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // Get the absolute path of the raw/***.bc file. 562b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao 572b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // Generate the absolute path. This doesn't do everything it 582b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // should, e.g. if resName is "./out/whatever" it doesn't crunch 592b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // the leading "./" out because this if-block is not triggered, 602b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // but it'll make do. 612b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // 622b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao if (getcwd(cacheFile, kBufLen) == NULL) { 632b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao LOGE("Can't get CWD while opening raw/***.bc file\n"); 642b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao return NULL; 652b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao } 662b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // Append "/" at the end of cacheFile so far. 672b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao strncat(cacheFile, "/", kBufLen); 682b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao } 692b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao 702b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // cacheFile = resName + extName 712b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // 722b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao strncat(cacheFile, resName, kBufLen); 732b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao if (extName != NULL) { 742b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // TODO(srhines): strncat() is a bit dangerous 752b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao strncat(cacheFile, extName, kBufLen); 762b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao } 772b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao 782b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // Turn the path into a flat filename by replacing 792b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // any slashes after the first one with '@' characters. 802b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao char *cp = cacheFile + 1; 812b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao while (*cp != '\0') { 822b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao if (*cp == '/') { 832b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao *cp = '@'; 842b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao } 852b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao cp++; 862b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao } 872b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao 882b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao // Tack on the file name for the actual cache file path. 892b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao strncpy(cachePath, cacheDir, kBufLen); 902b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao strncat(cachePath, cacheFile, kBufLen); 912b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao 922b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao LOGV("Cache file for '%s' '%s' is '%s'\n", resName, extName, cachePath); 932b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao return strdup(cachePath); 942b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao} 95326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 96afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukScriptC::ScriptC(Context *rsc) : Script(rsc) { 97741aac95b777b2e6cb90f484a05e489a79a6ef05Jason Sams LOGD(">>>> ScriptC ctor called, obj=%p", this); 98be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams mBccScript = NULL; 99efb8de1ef851c9c2a042ad06f64e33bb8b366041Jason Sams memset(&mProgram, 0, sizeof(mProgram)); 100326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 101326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 102afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukScriptC::~ScriptC() { 103741aac95b777b2e6cb90f484a05e489a79a6ef05Jason Sams LOGD(">>>> ~ScriptC() mBccScript = %p", mBccScript); 104be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams if (mBccScript) { 1052b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao bccDisposeScript(mBccScript); 106a6ebec8cf4a9c420fba74a80c7238b5f20b5966aShih-wei Liao LOGD(">>>> ~ScriptC(mBCCScript)"); 1071ef8b80bc49f415b93912fe7335f852953bde4a4Jack Palevich } 108e402ed33486730f1d06f080cdfc48132bf612b3aJason Sams free(mEnviroment.mScriptText); 109e402ed33486730f1d06f080cdfc48132bf612b3aJason Sams mEnviroment.mScriptText = NULL; 110326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 111326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 112afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid ScriptC::setupScript(Context *rsc) { 113c61346b91434307c5003029017b54ce9c49112beJason Sams mEnviroment.mStartTimeMillis 114c61346b91434307c5003029017b54ce9c49112beJason Sams = nanoseconds_to_milliseconds(systemTime(SYSTEM_TIME_MONOTONIC)); 115c61346b91434307c5003029017b54ce9c49112beJason Sams 116be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams for (uint32_t ct=0; ct < mEnviroment.mFieldCount; ct++) { 117900f1616bf33c7ba13cf2a737832a95bcd176388Jason Sams if (mSlots[ct].get() && !mTypes[ct].get()) { 118900f1616bf33c7ba13cf2a737832a95bcd176388Jason Sams mTypes[ct].set(mSlots[ct]->getType()); 119900f1616bf33c7ba13cf2a737832a95bcd176388Jason Sams } 120900f1616bf33c7ba13cf2a737832a95bcd176388Jason Sams 121900f1616bf33c7ba13cf2a737832a95bcd176388Jason Sams if (!mTypes[ct].get()) 122be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams continue; 123900f1616bf33c7ba13cf2a737832a95bcd176388Jason Sams void *ptr = NULL; 124900f1616bf33c7ba13cf2a737832a95bcd176388Jason Sams if (mSlots[ct].get()) { 125900f1616bf33c7ba13cf2a737832a95bcd176388Jason Sams ptr = mSlots[ct]->getPtr(); 126900f1616bf33c7ba13cf2a737832a95bcd176388Jason Sams } 127be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams void **dest = ((void ***)mEnviroment.mFieldAddress)[ct]; 128be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams 129b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams if (rsc->props.mLogScripts) { 1302fad7e47a8177235bb88fe271572bd986a7e4136Jason Sams if (mSlots[ct].get() != NULL) { 1312fad7e47a8177235bb88fe271572bd986a7e4136Jason Sams LOGV("%p ScriptC::setupScript slot=%i dst=%p src=%p type=%p", rsc, ct, dest, ptr, mSlots[ct]->getType()); 1322fad7e47a8177235bb88fe271572bd986a7e4136Jason Sams } else { 1332fad7e47a8177235bb88fe271572bd986a7e4136Jason Sams LOGV("%p ScriptC::setupScript slot=%i dst=%p src=%p type=null", rsc, ct, dest, ptr); 1342fad7e47a8177235bb88fe271572bd986a7e4136Jason Sams } 135b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams } 136be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams 137be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams if (dest) { 138be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams *dest = ptr; 139ada7f272890d8791bc518c95989ad7d13050834dJason Sams } 140ada7f272890d8791bc518c95989ad7d13050834dJason Sams } 141ada7f272890d8791bc518c95989ad7d13050834dJason Sams} 142ada7f272890d8791bc518c95989ad7d13050834dJason Sams 143afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukconst Allocation *ScriptC::ptrToAllocation(const void *ptr) const { 144ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams if (!ptr) { 145ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams return NULL; 146ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams } 147ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams for (uint32_t ct=0; ct < mEnviroment.mFieldCount; ct++) { 148ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams if (!mSlots[ct].get()) 149ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams continue; 150ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams if (mSlots[ct]->getPtr() == ptr) { 151ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams return mSlots[ct].get(); 152ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams } 153ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams } 154ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams LOGE("ScriptC::ptrToAllocation, failed to find %p", ptr); 155ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams return NULL; 156ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams} 157ce92d4baf7a5bce097228fdd4498601764cd4014Jason Sams 158afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukScript * ScriptC::setTLS(Script *sc) { 15922fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams Context::ScriptTLSStruct * tls = (Context::ScriptTLSStruct *) 16022fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams pthread_getspecific(Context::gThreadTLSKey); 16122fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams rsAssert(tls); 162c61346b91434307c5003029017b54ce9c49112beJason Sams Script *old = tls->mScript; 163c61346b91434307c5003029017b54ce9c49112beJason Sams tls->mScript = sc; 164c61346b91434307c5003029017b54ce9c49112beJason Sams return old; 16522fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams} 16622fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams 167afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid ScriptC::setupGLState(Context *rsc) { 168a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams if (mEnviroment.mFragmentStore.get()) { 16960709257bbdeb0c50f39b9c8969dc76264d6e142Jason Sams rsc->setProgramStore(mEnviroment.mFragmentStore.get()); 170a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams } 171a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams if (mEnviroment.mFragment.get()) { 17260709257bbdeb0c50f39b9c8969dc76264d6e142Jason Sams rsc->setProgramFragment(mEnviroment.mFragment.get()); 173a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams } 1748ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams if (mEnviroment.mVertex.get()) { 17560709257bbdeb0c50f39b9c8969dc76264d6e142Jason Sams rsc->setProgramVertex(mEnviroment.mVertex.get()); 1768ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams } 177b681c8af69ef6938c80dd5cc5fc920db94ee9f43Jason Sams if (mEnviroment.mRaster.get()) { 17860709257bbdeb0c50f39b9c8969dc76264d6e142Jason Sams rsc->setProgramRaster(mEnviroment.mRaster.get()); 179b681c8af69ef6938c80dd5cc5fc920db94ee9f43Jason Sams } 180c61346b91434307c5003029017b54ce9c49112beJason Sams} 181a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams 182afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukuint32_t ScriptC::run(Context *rsc) { 183c61346b91434307c5003029017b54ce9c49112beJason Sams if (mProgram.mRoot == NULL) { 184c61346b91434307c5003029017b54ce9c49112beJason Sams rsc->setError(RS_ERROR_BAD_SCRIPT, "Attempted to run bad script"); 185c61346b91434307c5003029017b54ce9c49112beJason Sams return 0; 1869c4e4ca9ff75a7fe18544c83fcf782e46c9b6ac2Joe Onorato } 187c61346b91434307c5003029017b54ce9c49112beJason Sams 1881f24db404b2cf8768bf2bd53a3caeb5812b8ce9fJason Sams setupGLState(rsc); 189c61346b91434307c5003029017b54ce9c49112beJason Sams setupScript(rsc); 1901d54f10f3c23e0d7ec57e52ec3b0701a2a5ed24eJason Sams 1912dca84dd6c07992f78ad050177975f16486dd77eJason Sams uint32_t ret = 0; 192c61346b91434307c5003029017b54ce9c49112beJason Sams Script * oldTLS = setTLS(this); 193b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams 194b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams if (rsc->props.mLogScripts) { 195b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams LOGV("%p ScriptC::run invoking root, ptr %p", rsc, mProgram.mRoot); 196b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams } 197b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams 198be36bf3a76481737a7fa606a04144ceef80eb4f2Jason Sams ret = mProgram.mRoot(); 199b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams 200b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams if (rsc->props.mLogScripts) { 201b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams LOGV("%p ScriptC::run invoking complete, ret=%i", rsc, ret); 202b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams } 203b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams 204c61346b91434307c5003029017b54ce9c49112beJason Sams setTLS(oldTLS); 205e45ac6e91864f2a6617b6a2c0aa87cdb62cf58fdJason Sams return ret; 206326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 207326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 2087bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Samstypedef struct { 2097bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams Context *rsc; 2107bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams ScriptC *script; 2117bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams const Allocation * ain; 2127bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams Allocation * aout; 2137bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams const void * usr; 2147bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 2157bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t mSliceSize; 2167bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams volatile int mSliceNum; 2177bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 2187bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams const uint8_t *ptrIn; 2197bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t eStrideIn; 2207bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint8_t *ptrOut; 2217bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t eStrideOut; 2227bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 2237bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t xStart; 2247bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t xEnd; 2257bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t yStart; 2267bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t yEnd; 2277bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t zStart; 2287bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t zEnd; 2297bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t arrayStart; 2307bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t arrayEnd; 2317bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 2327bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t dimX; 2337bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t dimY; 2347bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t dimZ; 2357bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t dimArray; 2367bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams} MTLaunchStruct; 2377bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Samstypedef int (*rs_t)(const void *, void *, const void *, uint32_t, uint32_t, uint32_t, uint32_t); 2387bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 239afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukstatic void wc_xy(void *usr, uint32_t idx) { 2407bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams MTLaunchStruct *mtls = (MTLaunchStruct *)usr; 2417bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 2427bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams while (1) { 2437bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t slice = (uint32_t)android_atomic_inc(&mtls->mSliceNum); 2447bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t yStart = mtls->yStart + slice * mtls->mSliceSize; 2457bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t yEnd = yStart + mtls->mSliceSize; 2467bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams yEnd = rsMin(yEnd, mtls->yEnd); 2477bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams if (yEnd <= yStart) { 2487bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams return; 2497bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams } 2507bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 2517bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams //LOGE("usr idx %i, x %i,%i y %i,%i", idx, mtls->xStart, mtls->xEnd, yStart, yEnd); 2528d957fa762eff6c03a93ddea9405d9575665f1ecJason Sams //LOGE("usr ptr in %p, out %p", mtls->ptrIn, mtls->ptrOut); 2537bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams for (uint32_t y = yStart; y < yEnd; y++) { 2547bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint32_t offset = mtls->dimX * y; 2557bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams uint8_t *xPtrOut = mtls->ptrOut + (mtls->eStrideOut * offset); 2567bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams const uint8_t *xPtrIn = mtls->ptrIn + (mtls->eStrideIn * offset); 2577bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 2587bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams for (uint32_t x = mtls->xStart; x < mtls->xEnd; x++) { 2597bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams ((rs_t)mtls->script->mProgram.mRoot) (xPtrIn, xPtrOut, mtls->usr, x, y, 0, 0); 2607bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams xPtrIn += mtls->eStrideIn; 2617bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams xPtrOut += mtls->eStrideOut; 2627bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams } 2637bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams } 2647bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams } 2657bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams} 2667bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 267afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukstatic void wc_x(void *usr, uint32_t idx) { 268177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams MTLaunchStruct *mtls = (MTLaunchStruct *)usr; 269177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams 270177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams while (1) { 271177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams uint32_t slice = (uint32_t)android_atomic_inc(&mtls->mSliceNum); 272177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams uint32_t xStart = mtls->xStart + slice * mtls->mSliceSize; 273177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams uint32_t xEnd = xStart + mtls->mSliceSize; 274177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams xEnd = rsMin(xEnd, mtls->xEnd); 275177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams if (xEnd <= xStart) { 276177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams return; 277177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams } 278177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams 279177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams //LOGE("usr idx %i, x %i,%i y %i,%i", idx, mtls->xStart, mtls->xEnd, yStart, yEnd); 280177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams //LOGE("usr ptr in %p, out %p", mtls->ptrIn, mtls->ptrOut); 281177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams uint8_t *xPtrOut = mtls->ptrOut + (mtls->eStrideOut * xStart); 282177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams const uint8_t *xPtrIn = mtls->ptrIn + (mtls->eStrideIn * xStart); 283177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams for (uint32_t x = xStart; x < xEnd; x++) { 284177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams ((rs_t)mtls->script->mProgram.mRoot) (xPtrIn, xPtrOut, mtls->usr, x, 0, 0, 0); 285177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams xPtrIn += mtls->eStrideIn; 286177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams xPtrOut += mtls->eStrideOut; 287177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams } 288177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams } 289177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams} 290177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams 291ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Samsvoid ScriptC::runForEach(Context *rsc, 292ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams const Allocation * ain, 293ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams Allocation * aout, 294ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams const void * usr, 295afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchouk const RsScriptCall *sc) { 2967bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams MTLaunchStruct mtls; 2977bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams memset(&mtls, 0, sizeof(mtls)); 29860709257bbdeb0c50f39b9c8969dc76264d6e142Jason Sams Context::PushState ps(rsc); 2997bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 3007bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams if (ain) { 3017bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.dimX = ain->getType()->getDimX(); 3027bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.dimY = ain->getType()->getDimY(); 3037bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.dimZ = ain->getType()->getDimZ(); 3047bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams //mtls.dimArray = ain->getType()->getDimArray(); 3057bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams } else if (aout) { 3067bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.dimX = aout->getType()->getDimX(); 3077bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.dimY = aout->getType()->getDimY(); 3087bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.dimZ = aout->getType()->getDimZ(); 3097bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams //mtls.dimArray = aout->getType()->getDimArray(); 3107bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams } else { 3117bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams rsc->setError(RS_ERROR_BAD_SCRIPT, "rsForEach called with null allocations"); 3127bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams return; 3137bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams } 314ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams 315ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams if (!sc || (sc->xEnd == 0)) { 3167bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.xEnd = mtls.dimX; 317ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams } else { 3187bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams rsAssert(sc->xStart < mtls.dimX); 3197bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams rsAssert(sc->xEnd <= mtls.dimX); 320ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams rsAssert(sc->xStart < sc->xEnd); 3217bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.xStart = rsMin(mtls.dimX, sc->xStart); 3227bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.xEnd = rsMin(mtls.dimX, sc->xEnd); 3237bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams if (mtls.xStart >= mtls.xEnd) return; 324ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams } 325c61346b91434307c5003029017b54ce9c49112beJason Sams 326ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams if (!sc || (sc->yEnd == 0)) { 3277bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.yEnd = mtls.dimY; 328ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams } else { 3297bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams rsAssert(sc->yStart < mtls.dimY); 3307bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams rsAssert(sc->yEnd <= mtls.dimY); 331ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams rsAssert(sc->yStart < sc->yEnd); 3327bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.yStart = rsMin(mtls.dimY, sc->yStart); 3337bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.yEnd = rsMin(mtls.dimY, sc->yEnd); 3347bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams if (mtls.yStart >= mtls.yEnd) return; 335ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams } 336ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams 3377bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.xEnd = rsMax((uint32_t)1, mtls.xEnd); 3387bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.yEnd = rsMax((uint32_t)1, mtls.yEnd); 3397bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.zEnd = rsMax((uint32_t)1, mtls.zEnd); 3407bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.arrayEnd = rsMax((uint32_t)1, mtls.arrayEnd); 341ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams 342ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams rsAssert(ain->getType()->getDimZ() == 0); 343c61346b91434307c5003029017b54ce9c49112beJason Sams 3441f24db404b2cf8768bf2bd53a3caeb5812b8ce9fJason Sams setupGLState(rsc); 345c61346b91434307c5003029017b54ce9c49112beJason Sams setupScript(rsc); 346c61346b91434307c5003029017b54ce9c49112beJason Sams Script * oldTLS = setTLS(this); 347c61346b91434307c5003029017b54ce9c49112beJason Sams 3487bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.rsc = rsc; 3497bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.ain = ain; 3507bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.aout = aout; 3517bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.script = this; 3527bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.usr = usr; 3537bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.mSliceSize = 10; 3547bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.mSliceNum = 0; 3557bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 3567bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.ptrIn = NULL; 3577bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.eStrideIn = 0; 3587bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams if (ain) { 3597bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.ptrIn = (const uint8_t *)ain->getPtr(); 3607bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.eStrideIn = ain->getType()->getElementSizeBytes(); 3617bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams } 362c61346b91434307c5003029017b54ce9c49112beJason Sams 3637bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.ptrOut = NULL; 3647bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.eStrideOut = 0; 365c61346b91434307c5003029017b54ce9c49112beJason Sams if (aout) { 3667bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.ptrOut = (uint8_t *)aout->getPtr(); 3677bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams mtls.eStrideOut = aout->getType()->getElementSizeBytes(); 368c61346b91434307c5003029017b54ce9c49112beJason Sams } 369c61346b91434307c5003029017b54ce9c49112beJason Sams 370177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams if ((rsc->getWorkerPoolSize() > 1) && mEnviroment.mIsThreadable) { 371177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams if (mtls.dimY > 1) { 372177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams rsc->launchThreads(wc_xy, &mtls); 373177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams } else { 374177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams rsc->launchThreads(wc_x, &mtls); 375177f8446d58e5b1a4258935371a9450dbe34dca6Jason Sams } 3767bf29ddc35450d8064541c42c99a1f48be6cf0ddJason Sams 377181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams //LOGE("launch 1"); 378181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams } else { 3798d957fa762eff6c03a93ddea9405d9575665f1ecJason Sams //LOGE("launch 3"); 380181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams for (uint32_t ar = mtls.arrayStart; ar < mtls.arrayEnd; ar++) { 381181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams for (uint32_t z = mtls.zStart; z < mtls.zEnd; z++) { 382181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams for (uint32_t y = mtls.yStart; y < mtls.yEnd; y++) { 383181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams uint32_t offset = mtls.dimX * mtls.dimY * mtls.dimZ * ar + 384181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams mtls.dimX * mtls.dimY * z + 385181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams mtls.dimX * y; 386181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams uint8_t *xPtrOut = mtls.ptrOut + (mtls.eStrideOut * offset); 387181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams const uint8_t *xPtrIn = mtls.ptrIn + (mtls.eStrideIn * offset); 388181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams 389181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams for (uint32_t x = mtls.xStart; x < mtls.xEnd; x++) { 390181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams ((rs_t)mProgram.mRoot) (xPtrIn, xPtrOut, usr, x, y, z, ar); 391181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams xPtrIn += mtls.eStrideIn; 392181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams xPtrOut += mtls.eStrideOut; 393181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams } 394ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams } 395ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams } 396ace3e01f07252ee0fa47fcf1ac48864d8220b90eJason Sams } 397c61346b91434307c5003029017b54ce9c49112beJason Sams } 398181334092f78273d0da2d854ebbe7b24508d3bfcJason Sams 399c61346b91434307c5003029017b54ce9c49112beJason Sams setTLS(oldTLS); 400c61346b91434307c5003029017b54ce9c49112beJason Sams} 401c61346b91434307c5003029017b54ce9c49112beJason Sams 402afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid ScriptC::Invoke(Context *rsc, uint32_t slot, const void *data, uint32_t len) { 40322fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams if ((slot >= mEnviroment.mInvokeFunctionCount) || 40422fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams (mEnviroment.mInvokeFunctions[slot] == NULL)) { 40522fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams rsc->setError(RS_ERROR_BAD_SCRIPT, "Calling invoke on bad script"); 40622fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams return; 40722fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams } 408c61346b91434307c5003029017b54ce9c49112beJason Sams setupScript(rsc); 409c61346b91434307c5003029017b54ce9c49112beJason Sams Script * oldTLS = setTLS(this); 41022fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams 411b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams if (rsc->props.mLogScripts) { 412b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams LOGV("%p ScriptC::Invoke invoking slot %i, ptr %p", rsc, slot, mEnviroment.mInvokeFunctions[slot]); 413b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams } 4142a63bf6c293d89c8e3725cfb7ee2add3dd3f0246Jason Sams ((void (*)(const void *, uint32_t)) 4152a63bf6c293d89c8e3725cfb7ee2add3dd3f0246Jason Sams mEnviroment.mInvokeFunctions[slot])(data, len); 416b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams if (rsc->props.mLogScripts) { 417b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams LOGV("%p ScriptC::Invoke complete", rsc); 418b9077f48e2c5aff1838bbbeccf56c2b5f5490ab6Jason Sams } 4192a63bf6c293d89c8e3725cfb7ee2add3dd3f0246Jason Sams 420c61346b91434307c5003029017b54ce9c49112beJason Sams setTLS(oldTLS); 42122fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams} 42222fa371bf64b8aae786acfe5d711af21afb13a93Jason Sams 423afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukScriptCState::ScriptCState() { 42401b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines mScript.clear(); 425326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 426326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 427afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukScriptCState::~ScriptCState() { 42801b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines mScript.clear(); 429326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 430326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 431afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid ScriptCState::init(Context *rsc) { 43201b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines clear(rsc); 43301b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines} 43401b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines 435afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid ScriptCState::clear(Context *rsc) { 43601b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines rsAssert(rsc); 43701b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines mScript.clear(); 43801b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines mScript.set(new ScriptC(rsc)); 4391f52633fefd1862451e6a30209d590680d02beecJason Sams} 4401f52633fefd1862451e6a30209d590680d02beecJason Sams 4412b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liaostatic void* symbolLookup(void* pContext, char const* name) { 442aeb094b520d8ea49b74129927578f18f758c873eJason Sams const ScriptCState::SymbolTable_t *sym; 443dd663fa8367bfacb6c77b368f91adf614cd0bebaJason Sams ScriptC *s = (ScriptC *)pContext; 444f17b13be814135b85e6aed7e3cca441c8e00e8c4Shih-wei Liao if (!strcmp(name, "__isThreadable")) { 4452b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao return (void*) s->mEnviroment.mIsThreadable; 446f17b13be814135b85e6aed7e3cca441c8e00e8c4Shih-wei Liao } else if (!strcmp(name, "__clearThreadable")) { 447f17b13be814135b85e6aed7e3cca441c8e00e8c4Shih-wei Liao s->mEnviroment.mIsThreadable = false; 448f17b13be814135b85e6aed7e3cca441c8e00e8c4Shih-wei Liao return NULL; 449f17b13be814135b85e6aed7e3cca441c8e00e8c4Shih-wei Liao } 450aeb094b520d8ea49b74129927578f18f758c873eJason Sams sym = ScriptCState::lookupSymbol(name); 4516bfc1b91dc7684f8ad9dc9f605e92ed8a31d5bbeJason Sams if (!sym) { 4526bfc1b91dc7684f8ad9dc9f605e92ed8a31d5bbeJason Sams sym = ScriptCState::lookupSymbolCL(name); 453aeb094b520d8ea49b74129927578f18f758c873eJason Sams } 4546bfc1b91dc7684f8ad9dc9f605e92ed8a31d5bbeJason Sams if (!sym) { 4556bfc1b91dc7684f8ad9dc9f605e92ed8a31d5bbeJason Sams sym = ScriptCState::lookupSymbolGL(name); 456aeb094b520d8ea49b74129927578f18f758c873eJason Sams } 45729df66f82aeef7fa7e2cf00edbf00d43c822b05aJason Sams if (sym) { 4586bfc1b91dc7684f8ad9dc9f605e92ed8a31d5bbeJason Sams s->mEnviroment.mIsThreadable &= sym->threadable; 45929df66f82aeef7fa7e2cf00edbf00d43c822b05aJason Sams return sym->mPtr; 46029df66f82aeef7fa7e2cf00edbf00d43c822b05aJason Sams } 46129df66f82aeef7fa7e2cf00edbf00d43c822b05aJason Sams LOGE("ScriptC sym lookup failed for %s", name); 46229df66f82aeef7fa7e2cf00edbf00d43c822b05aJason Sams return NULL; 46329df66f82aeef7fa7e2cf00edbf00d43c822b05aJason Sams} 464a4a54e42fc710a62b47cbcb9d64c34a190429d9eJason Sams 46580761ecd7b19a44e6bb77437f4d99ee6b58fe251Shih-wei Liao#if 0 4661f9ba735cc429ae8df140644f6f0a4481f2a8068Shih-wei Liaoextern const char rs_runtime_lib_bc[]; 4671f9ba735cc429ae8df140644f6f0a4481f2a8068Shih-wei Liaoextern unsigned rs_runtime_lib_bc_size; 46880761ecd7b19a44e6bb77437f4d99ee6b58fe251Shih-wei Liao#endif 4691f9ba735cc429ae8df140644f6f0a4481f2a8068Shih-wei Liao 47026b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Samsbool ScriptCState::runCompiler(Context *rsc, 471ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei Liao ScriptC *s, 472ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei Liao const char *resName, 473ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei Liao const char *cacheDir) { 4747b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mBccScript = bccCreateScript(); 47537150def08fc971e8422245efef27265331cace4Shih-wei Liao 4767b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mEnviroment.mIsThreadable = true; 47737150def08fc971e8422245efef27265331cace4Shih-wei Liao 4787b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines bccRegisterSymbolCallback(s->mBccScript, symbolLookup, s); 47937150def08fc971e8422245efef27265331cace4Shih-wei Liao 4807b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (bccReadBC(s->mBccScript, 4817b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines resName, 4827b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mEnviroment.mScriptText, 4837b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mEnviroment.mScriptTextLength, 0) != 0) { 4847b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines LOGE("bcc: FAILS to read bitcode"); 48526b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return false; 4867b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 48737150def08fc971e8422245efef27265331cace4Shih-wei Liao 4885439184e13077059e4caffbdadfcfe37ec99ec10Shih-wei Liao#if 1 4897b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (bccLinkBC(s->mBccScript, 4907b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines resName, 4917b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines NULL /*rs_runtime_lib_bc*/, 4927b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines 1 /*rs_runtime_lib_bc_size*/ 4937b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines /*"1" means skip buffer here, and let libbcc decide*/, 4947b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines 0) != 0) { 4957b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines LOGE("bcc: FAILS to link bitcode"); 49626b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return false; 4977b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 49837150def08fc971e8422245efef27265331cace4Shih-wei Liao#endif 4997b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines char *cachePath = genCacheFileName(cacheDir, resName, ".oBCC"); 50037150def08fc971e8422245efef27265331cace4Shih-wei Liao 5017b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (bccPrepareExecutable(s->mBccScript, cachePath, 0) != 0) { 5027b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines LOGE("bcc: FAILS to prepare executable"); 50326b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return false; 5047b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 50537150def08fc971e8422245efef27265331cace4Shih-wei Liao 5067b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines free(cachePath); 5072b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao 5087b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mProgram.mRoot = reinterpret_cast<int (*)()>(bccGetFuncAddr(s->mBccScript, "root")); 5097b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mProgram.mInit = reinterpret_cast<void (*)()>(bccGetFuncAddr(s->mBccScript, "init")); 510f1685045e3423f2e342c3d74c206a25d72eba89dJason Sams 5118c6bc6930d4996b897de8e5113da4a4efae2310cJason Sams if (s->mProgram.mInit) { 5128c6bc6930d4996b897de8e5113da4a4efae2310cJason Sams s->mProgram.mInit(); 5131d54f10f3c23e0d7ec57e52ec3b0701a2a5ed24eJason Sams } 5141d54f10f3c23e0d7ec57e52ec3b0701a2a5ed24eJason Sams 5152b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao s->mEnviroment.mInvokeFunctionCount = bccGetExportFuncCount(s->mBccScript); 516afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchouk if (s->mEnviroment.mInvokeFunctionCount <= 0) 5178c88090e8cccab103eebe2ff569e116e9f5fb208Jason Sams s->mEnviroment.mInvokeFunctions = NULL; 5188c88090e8cccab103eebe2ff569e116e9f5fb208Jason Sams else { 5198c88090e8cccab103eebe2ff569e116e9f5fb208Jason Sams s->mEnviroment.mInvokeFunctions = (Script::InvokeFunc_t*) calloc(s->mEnviroment.mInvokeFunctionCount, sizeof(Script::InvokeFunc_t)); 5202b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao bccGetExportFuncList(s->mBccScript, s->mEnviroment.mInvokeFunctionCount, (void **) s->mEnviroment.mInvokeFunctions); 5218c88090e8cccab103eebe2ff569e116e9f5fb208Jason Sams } 5228c88090e8cccab103eebe2ff569e116e9f5fb208Jason Sams 5232b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao s->mEnviroment.mFieldCount = bccGetExportVarCount(s->mBccScript); 524afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchouk if (s->mEnviroment.mFieldCount <= 0) 525a226b166dde77ae4bf4202a60afd14339b040dd3Shih-wei Liao s->mEnviroment.mFieldAddress = NULL; 526a226b166dde77ae4bf4202a60afd14339b040dd3Shih-wei Liao else { 527a226b166dde77ae4bf4202a60afd14339b040dd3Shih-wei Liao s->mEnviroment.mFieldAddress = (void **) calloc(s->mEnviroment.mFieldCount, sizeof(void *)); 5282b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao bccGetExportVarList(s->mBccScript, s->mEnviroment.mFieldCount, (void **) s->mEnviroment.mFieldAddress); 529700ba38f022208686523ab4280c4fc9f102aa273Alex Sakhartchouk s->initSlots(); 530a226b166dde77ae4bf4202a60afd14339b040dd3Shih-wei Liao } 5311d54f10f3c23e0d7ec57e52ec3b0701a2a5ed24eJason Sams 5328c6bc6930d4996b897de8e5113da4a4efae2310cJason Sams s->mEnviroment.mFragment.set(rsc->getDefaultProgramFragment()); 5338c6bc6930d4996b897de8e5113da4a4efae2310cJason Sams s->mEnviroment.mVertex.set(rsc->getDefaultProgramVertex()); 534ccc010bb7c0f89e162bf60033968a20be90a903aJason Sams s->mEnviroment.mFragmentStore.set(rsc->getDefaultProgramStore()); 535b681c8af69ef6938c80dd5cc5fc920db94ee9f43Jason Sams s->mEnviroment.mRaster.set(rsc->getDefaultProgramRaster()); 536a4a54e42fc710a62b47cbcb9d64c34a190429d9eJason Sams 5377b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines const static int pragmaMax = 16; 5387b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines size_t pragmaCount = bccGetPragmaCount(s->mBccScript); 5397b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines char const *keys[pragmaMax]; 5407b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines char const *values[pragmaMax]; 5417b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines bccGetPragmaList(s->mBccScript, pragmaMax, keys, values); 5422b2e62117780d89bcd38f3ae453c4105a360ff79Shih-wei Liao 5437b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines for (size_t i=0; i < pragmaCount; ++i) { 5447b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines //LOGE("pragma %s %s", keys[i], values[i]); 5457b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(keys[i], "version")) { 546b5dc6af59c717fd871874930ac8e7b2b0f26dae2Stephen Hines if (!strcmp(values[i], "1")) { 547b5dc6af59c717fd871874930ac8e7b2b0f26dae2Stephen Hines continue; 548b5dc6af59c717fd871874930ac8e7b2b0f26dae2Stephen Hines } 549b5dc6af59c717fd871874930ac8e7b2b0f26dae2Stephen Hines LOGE("Invalid version pragma value: %s\n", values[i]); 55026b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return false; 5517b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 5527b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines 5537b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(keys[i], "stateVertex")) { 5547b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(values[i], "default")) { 5551030893d9b99b72468034da13df025bda479bb97Jason Sams continue; 5561030893d9b99b72468034da13df025bda479bb97Jason Sams } 5577b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(values[i], "parent")) { 5587b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mEnviroment.mVertex.clear(); 5597b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines continue; 5601030893d9b99b72468034da13df025bda479bb97Jason Sams } 561b5dc6af59c717fd871874930ac8e7b2b0f26dae2Stephen Hines LOGE("Unrecognized value %s passed to stateVertex", values[i]); 56226b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return false; 5637b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 5641030893d9b99b72468034da13df025bda479bb97Jason Sams 5657b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(keys[i], "stateRaster")) { 5667b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(values[i], "default")) { 5677b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines continue; 5681030893d9b99b72468034da13df025bda479bb97Jason Sams } 5697b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(values[i], "parent")) { 5707b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mEnviroment.mRaster.clear(); 5717b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines continue; 5727b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 573b5dc6af59c717fd871874930ac8e7b2b0f26dae2Stephen Hines LOGE("Unrecognized value %s passed to stateRaster", values[i]); 57426b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return false; 5757b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 5761030893d9b99b72468034da13df025bda479bb97Jason Sams 5777b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(keys[i], "stateFragment")) { 5787b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(values[i], "default")) { 5797b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines continue; 5807b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 5817b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(values[i], "parent")) { 5827b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mEnviroment.mFragment.clear(); 5837b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines continue; 5841030893d9b99b72468034da13df025bda479bb97Jason Sams } 585b5dc6af59c717fd871874930ac8e7b2b0f26dae2Stephen Hines LOGE("Unrecognized value %s passed to stateFragment", values[i]); 58626b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return false; 5877b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 5881030893d9b99b72468034da13df025bda479bb97Jason Sams 5897b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(keys[i], "stateStore")) { 5907b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(values[i], "default")) { 5917b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines continue; 5921030893d9b99b72468034da13df025bda479bb97Jason Sams } 5937b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines if (!strcmp(values[i], "parent")) { 5947b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines s->mEnviroment.mFragmentStore.clear(); 5957b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines continue; 5967b337b1ce7e8861a254d7d06c2d0a26bced9b52bStephen Hines } 597b5dc6af59c717fd871874930ac8e7b2b0f26dae2Stephen Hines LOGE("Unrecognized value %s passed to stateStore", values[i]); 59826b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return false; 5991030893d9b99b72468034da13df025bda479bb97Jason Sams } 6001030893d9b99b72468034da13df025bda479bb97Jason Sams } 60126b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return true; 60257b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato} 6031030893d9b99b72468034da13df025bda479bb97Jason Sams 604326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsnamespace android { 605326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsnamespace renderscript { 606326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 607afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid rsi_ScriptCBegin(Context * rsc) { 608326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ScriptCState *ss = &rsc->mScriptC; 60901b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines ss->clear(rsc); 610326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 611326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 612afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid rsi_ScriptCSetText(Context *rsc, const char *text, uint32_t len) { 6131f52633fefd1862451e6a30209d590680d02beecJason Sams ScriptCState *ss = &rsc->mScriptC; 614e402ed33486730f1d06f080cdfc48132bf612b3aJason Sams 615e402ed33486730f1d06f080cdfc48132bf612b3aJason Sams char *t = (char *)malloc(len + 1); 616e402ed33486730f1d06f080cdfc48132bf612b3aJason Sams memcpy(t, text, len); 617e402ed33486730f1d06f080cdfc48132bf612b3aJason Sams t[len] = 0; 618e402ed33486730f1d06f080cdfc48132bf612b3aJason Sams ss->mScript->mEnviroment.mScriptText = t; 6198c6bc6930d4996b897de8e5113da4a4efae2310cJason Sams ss->mScript->mEnviroment.mScriptTextLength = len; 6201f52633fefd1862451e6a30209d590680d02beecJason Sams} 6211f52633fefd1862451e6a30209d590680d02beecJason Sams 622ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei Liao 623ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei LiaoRsScript rsi_ScriptCCreate(Context *rsc, 62437150def08fc971e8422245efef27265331cace4Shih-wei Liao const char *packageName /* deprecated */, 625ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei Liao const char *resName, 626ce8a079bd4d296f9f1a24d7a5808d57f71dfc2e4Shih-wei Liao const char *cacheDir) 6279503b66e5329703535f7cc71a0755e6f3b1e39a6Shih-wei Liao{ 628326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ScriptCState *ss = &rsc->mScriptC; 629326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 630225afd317e101a7be5fe02c0a86361146ea89f05Jason Sams ObjectBaseRef<ScriptC> s(ss->mScript); 63101b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines ss->mScript.clear(); 632225afd317e101a7be5fe02c0a86361146ea89f05Jason Sams s->incUserRef(); 6331f52633fefd1862451e6a30209d590680d02beecJason Sams 63426b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams if (!ss->runCompiler(rsc, s.get(), resName, cacheDir)) { 63526b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams // Error during compile, destroy s and return null. 63626b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams s->zeroUserRef(); 63726b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams return NULL; 63826b2c9f5b0d77b69fb5edf96dab7d57f1de1d594Jason Sams } 63901b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines ss->clear(rsc); 64001b7d2995f9cbd33a9ccdf861fe959743a4b9954Stephen Hines return s.get(); 641326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 642326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 643326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 644326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 645