rsContext.h revision fb03a22ad2adadc1ff50a8b50d43ad7fcc3fa6ed
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#ifndef ANDROID_RS_CONTEXT_H 18326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#define ANDROID_RS_CONTEXT_H 19326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 20a5577808737a394bdb156fbeb80018d0a4d8438dJason Sams#include "rsUtils.h" 21a5577808737a394bdb156fbeb80018d0a4d8438dJason Sams 22326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include <ui/Surface.h> 23326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 24fcd3192ebff8ab58d841836b7e94361d0998338cJason Sams#include "rsThreadIO.h" 25326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsType.h" 26326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsMatrix.h" 27326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsAllocation.h" 28e5ffb879ae535a899a486285a23bea05e912480fJason Sams#include "rsSimpleMesh.h" 29a89371c6f144b9049efe7689105feee2c4a38384Jason Sams#include "rsMesh.h" 30326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsDevice.h" 31326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsScriptC.h" 32326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsAllocation.h" 33326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsAdapter.h" 34326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsSampler.h" 35b5909ce06dd10dcb5ac715572a05b2d225b77c98Jason Sams#include "rsLight.h" 36326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsProgramFragment.h" 37326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsProgramFragmentStore.h" 385fd09d847586f9680b4f495413b6ca5fbb69af6eJason Sams#include "rsProgramRaster.h" 39326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsProgramVertex.h" 40326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 41326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsgApiStructs.h" 42326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsLocklessFifo.h" 43326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 44326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 45326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams// --------------------------------------------------------------------------- 46326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsnamespace android { 47326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsnamespace renderscript { 48326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 49cfb1d11ce6826fce7241d316d8b7dcab661f63a6Jason Samsclass Context 50326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 51326e0ddf89e8df2837752fbfd7a014814b32082cJason Samspublic: 52afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams Context(Device *, Surface *, bool useDepth); 53326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ~Context(); 54326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 55e57691037aea219562ac686429b4b98202aab7bcJason Sams static pthread_key_t gThreadTLSKey; 56fb03a22ad2adadc1ff50a8b50d43ad7fcc3fa6edJason Sams static uint32_t gThreadTLSKeyCount; 57fb03a22ad2adadc1ff50a8b50d43ad7fcc3fa6edJason Sams static pthread_mutex_t gInitMutex; 58fb03a22ad2adadc1ff50a8b50d43ad7fcc3fa6edJason Sams 59e57691037aea219562ac686429b4b98202aab7bcJason Sams struct ScriptTLSStruct { 60e57691037aea219562ac686429b4b98202aab7bcJason Sams Context * mContext; 61e57691037aea219562ac686429b4b98202aab7bcJason Sams Script * mScript; 62e57691037aea219562ac686429b4b98202aab7bcJason Sams }; 63e57691037aea219562ac686429b4b98202aab7bcJason Sams 64326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 65326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams //StructuredAllocationContext mStateAllocation; 66326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ElementState mStateElement; 67326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams TypeState mStateType; 68326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams SamplerState mStateSampler; 69326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ProgramFragmentState mStateFragment; 70326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ProgramFragmentStoreState mStateFragmentStore; 715fd09d847586f9680b4f495413b6ca5fbb69af6eJason Sams ProgramRasterState mStateRaster; 72326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ProgramVertexState mStateVertex; 7362bc1db27ba17f7eed0a6dfb639da7326753b268Jason Sams LightState mStateLight; 74326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 75326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ScriptCState mScriptC; 76326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 77326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams void swapBuffers(); 78326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams void setRootScript(Script *); 795fd09d847586f9680b4f495413b6ca5fbb69af6eJason Sams void setRaster(ProgramRaster *); 80326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams void setVertex(ProgramVertex *); 81326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams void setFragment(ProgramFragment *); 82326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams void setFragmentStore(ProgramFragmentStore *); 83326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 84326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams void updateSurface(void *sur); 85326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 86326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams const ProgramFragment * getFragment() {return mFragment.get();} 87326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams const ProgramFragmentStore * getFragmentStore() {return mFragmentStore.get();} 885fd09d847586f9680b4f495413b6ca5fbb69af6eJason Sams const ProgramRaster * getRaster() {return mRaster.get();} 89c9d43db4d216b01b13aebfdb31d5615909591b33Jason Sams const ProgramVertex * getVertex() {return mVertex.get();} 90326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 91326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams void setupCheck(); 92cfb1d11ce6826fce7241d316d8b7dcab661f63a6Jason Sams void allocationCheck(const Allocation *); 93326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 9486f1b23aaaf9b8822a009d8c3e585e46768abb6aJason Sams void pause(); 9586f1b23aaaf9b8822a009d8c3e585e46768abb6aJason Sams void resume(); 9686f1b23aaaf9b8822a009d8c3e585e46768abb6aJason Sams 97a4a54e42fc710a62b47cbcb9d64c34a190429d9eJason Sams void assignName(ObjectBase *obj, const char *name, uint32_t len); 98a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams void removeName(ObjectBase *obj); 99a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams ObjectBase * lookupName(const char *name) const; 100a4a54e42fc710a62b47cbcb9d64c34a190429d9eJason Sams void appendNameDefines(String8 *str) const; 10157b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato void appendVarDefines(String8 *str) const; 1028ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams 1038c401effb0837155fc39ca0364f57a882d127d38Jason Sams uint32_t getMessageToClient(void *data, size_t *receiveLen, size_t bufferLen, bool wait); 1048c401effb0837155fc39ca0364f57a882d127d38Jason Sams bool sendMessageToClient(void *data, uint32_t cmdID, size_t len, bool waitForSpace); 1053a27c952c013ad0a8e0c91bea76d895a07f7a56dJason Sams bool runScript(Script *s, uint32_t launchID); 1068c401effb0837155fc39ca0364f57a882d127d38Jason Sams 1078c401effb0837155fc39ca0364f57a882d127d38Jason Sams void initToClient(); 1088c401effb0837155fc39ca0364f57a882d127d38Jason Sams void deinitToClient(); 1098c401effb0837155fc39ca0364f57a882d127d38Jason Sams 1108ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams ProgramFragment * getDefaultProgramFragment() const { 1118ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams return mStateFragment.mDefault.get(); 1128ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams } 1138ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams ProgramVertex * getDefaultProgramVertex() const { 1148ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams return mStateVertex.mDefault.get(); 1158ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams } 1168ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams ProgramFragmentStore * getDefaultProgramFragmentStore() const { 1178ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams return mStateFragmentStore.mDefault.get(); 1188ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams } 1195fd09d847586f9680b4f495413b6ca5fbb69af6eJason Sams ProgramRaster * getDefaultProgramRaster() const { 1205fd09d847586f9680b4f495413b6ca5fbb69af6eJason Sams return mStateRaster.mDefault.get(); 1215fd09d847586f9680b4f495413b6ca5fbb69af6eJason Sams } 1228ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams 12357b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato void addInt32Define(const char* name, int32_t value) { 12457b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato mInt32Defines.add(String8(name), value); 12557b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato } 12657b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato 12757b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato void addFloatDefine(const char* name, float value) { 12857b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato mFloatDefines.add(String8(name), value); 12957b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato } 13057b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato 131afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams uint32_t getWidth() const {return mEGL.mWidth;} 132afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams uint32_t getHeight() const {return mEGL.mHeight;} 133e579df42e85d9e00f53c42ef1b78dbd209dba989Jason Sams 134fcd3192ebff8ab58d841836b7e94361d0998338cJason Sams 135fcd3192ebff8ab58d841836b7e94361d0998338cJason Sams ThreadIO mIO; 1365086938044e0a9b6b1138f915d0d252fe046e102Jason Sams void objDestroyAdd(ObjectBase *); 137fcd3192ebff8ab58d841836b7e94361d0998338cJason Sams 13824371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams // Timers 13924371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams enum Timers { 14024371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams RS_TIMER_IDLE, 14124371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams RS_TIMER_INTERNAL, 14224371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams RS_TIMER_SCRIPT, 14324371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams RS_TIMER_CLEAR_SWAP, 14424371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams _RS_TIMER_TOTAL 14524371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams }; 14624371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams uint64_t getTime() const; 14724371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams void timerInit(); 14824371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams void timerReset(); 14924371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams void timerSet(Timers); 15024371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams void timerPrint(); 1511d54f10f3c23e0d7ec57e52ec3b0701a2a5ed24eJason Sams void timerFrame(); 15224371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams 153afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams bool checkVersion1_1() const {return (mGL.mMajorVersion > 1) || (mGL.mMinorVersion >= 1); } 154afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams bool checkVersion2_0() const {return mGL.mMajorVersion >= 2; } 155afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams 1561fddd90849deaae89b546ff492c345d485bbce42Jason Sams struct { 1571fddd90849deaae89b546ff492c345d485bbce42Jason Sams bool mLogTimes; 1581fddd90849deaae89b546ff492c345d485bbce42Jason Sams bool mLogScripts; 1591fddd90849deaae89b546ff492c345d485bbce42Jason Sams bool mLogObjects; 1601fddd90849deaae89b546ff492c345d485bbce42Jason Sams } props; 16176371fff76412fd020e24ddb8bf1ddb5c75f0ed1Joe Onorato 162e514b45de8561fbc6ef6770845102ca10b0a69d7Jason Sams mutable const ObjectBase * mObjHead; 163e514b45de8561fbc6ef6770845102ca10b0a69d7Jason Sams 164326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsprotected: 165326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams Device *mDev; 166326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 167afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams struct { 168afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams EGLint mNumConfigs; 169afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams EGLint mMajorVersion; 170afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams EGLint mMinorVersion; 171afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams EGLConfig mConfig; 172afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams EGLContext mContext; 173afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams EGLSurface mSurface; 174afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams EGLint mWidth; 175afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams EGLint mHeight; 176afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams EGLDisplay mDisplay; 177afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams } mEGL; 178afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams 179afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams struct { 180afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams const uint8_t * mVendor; 181afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams const uint8_t * mRenderer; 182afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams const uint8_t * mVersion; 183afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams const uint8_t * mExtensions; 184afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams 185afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams uint32_t mMajorVersion; 186afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams uint32_t mMinorVersion; 187afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams 188afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams } mGL; 189326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 190326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams bool mRunning; 191326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams bool mExit; 192afcb25c65e8145d15aaf50a0ca38333954a97000Jason Sams bool mUseDepth; 19386f1b23aaaf9b8822a009d8c3e585e46768abb6aJason Sams bool mPaused; 194326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 195326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams pthread_t mThreadId; 196326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 197326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ObjectBaseRef<Script> mRootScript; 198326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ObjectBaseRef<ProgramFragment> mFragment; 199326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ObjectBaseRef<ProgramVertex> mVertex; 200326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ObjectBaseRef<ProgramFragmentStore> mFragmentStore; 2015fd09d847586f9680b4f495413b6ca5fbb69af6eJason Sams ObjectBaseRef<ProgramRaster> mRaster; 202326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 2035086938044e0a9b6b1138f915d0d252fe046e102Jason Sams 2045086938044e0a9b6b1138f915d0d252fe046e102Jason Sams struct ObjDestroyOOB { 2055086938044e0a9b6b1138f915d0d252fe046e102Jason Sams pthread_mutex_t mMutex; 2065086938044e0a9b6b1138f915d0d252fe046e102Jason Sams Vector<ObjectBase *> mDestroyList; 2075086938044e0a9b6b1138f915d0d252fe046e102Jason Sams bool mNeedToEmpty; 2085086938044e0a9b6b1138f915d0d252fe046e102Jason Sams }; 2095086938044e0a9b6b1138f915d0d252fe046e102Jason Sams ObjDestroyOOB mObjDestroy; 2105086938044e0a9b6b1138f915d0d252fe046e102Jason Sams bool objDestroyOOBInit(); 2115086938044e0a9b6b1138f915d0d252fe046e102Jason Sams void objDestroyOOBRun(); 2125086938044e0a9b6b1138f915d0d252fe046e102Jason Sams void objDestroyOOBDestroy(); 2135086938044e0a9b6b1138f915d0d252fe046e102Jason Sams 214326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsprivate: 215326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams Context(); 216326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 217326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams void initEGL(); 218326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 219a44cb29164726cd9d812117819abdd7b60dfdd93Jason Sams bool runRootScript(); 220326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 221326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams static void * threadProc(void *); 222326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 223326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams Surface *mWndSurface; 224a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams 225a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams Vector<ObjectBase *> mNames; 22657b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato KeyedVector<String8,int> mInt32Defines; 22757b79ceb1126e3797fa42367b97dd7bcfcda1ed9Joe Onorato KeyedVector<String8,float> mFloatDefines; 22824371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams 22924371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams uint64_t mTimers[_RS_TIMER_TOTAL]; 23024371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams Timers mTimerActive; 23124371d93cdb6999971c4058f78974da3c3d5fc64Jason Sams uint64_t mTimeLast; 2321d54f10f3c23e0d7ec57e52ec3b0701a2a5ed24eJason Sams uint64_t mTimeFrame; 2331d54f10f3c23e0d7ec57e52ec3b0701a2a5ed24eJason Sams uint64_t mTimeLastFrame; 234326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams}; 235326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 236326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 237326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 238326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#endif 239