rsProgramFragment.cpp revision 8ce125be69531dbf3a7e856d5e59d1b8e2789db0
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 "rsProgramFragment.h" 19326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 20326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsusing namespace android; 21326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsusing namespace android::renderscript; 22326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 23326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 24326e0ddf89e8df2837752fbfd7a014814b32082cJason SamsProgramFragment::ProgramFragment(Element *in, Element *out) : 25326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams Program(in, out) 26326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 27326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams for (uint32_t ct=0; ct < MAX_TEXTURE; ct++) { 28326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mEnvModes[ct] = RS_TEX_ENV_MODE_REPLACE; 29326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mTextureDimensions[ct] = 2; 30326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 31326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mTextureEnableMask = 0; 32326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 33326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 34326e0ddf89e8df2837752fbfd7a014814b32082cJason SamsProgramFragment::~ProgramFragment() 35326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 36326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 37326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 38326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid ProgramFragment::setupGL() 39326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 40326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams for (uint32_t ct=0; ct < MAX_TEXTURE; ct++) { 41326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams glActiveTexture(GL_TEXTURE0 + ct); 42326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (!(mTextureEnableMask & (1 << ct)) || 436678e9b2568ad041429a2477177133fe4932159fJason Sams //!mSamplers[ct].get() || 44326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams !mTextures[ct].get()) { 45326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 46326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams glDisable(GL_TEXTURE_2D); 47326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams continue; 48326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 49326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 50326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams glEnable(GL_TEXTURE_2D); 51326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams glBindTexture(GL_TEXTURE_2D, mTextures[ct]->getTextureID()); 52326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 53326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams switch(mEnvModes[ct]) { 54326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams case RS_TEX_ENV_MODE_REPLACE: 55326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams glTexEnvf(GL_TEXTURE_2D, GL_TEXTURE_ENV_MODE, GL_REPLACE); 56326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams break; 57326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams case RS_TEX_ENV_MODE_MODULATE: 58326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams glTexEnvf(GL_TEXTURE_2D, GL_TEXTURE_ENV_MODE, GL_MODULATE); 59326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams break; 60326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams case RS_TEX_ENV_MODE_DECAL: 61326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams glTexEnvf(GL_TEXTURE_2D, GL_TEXTURE_ENV_MODE, GL_DECAL); 62326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams break; 63326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 64326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 6539c8bc7be5751ec52693d21abdf139c4dfd29a2cJason Sams if (mSamplers[ct].get()) { 6639c8bc7be5751ec52693d21abdf139c4dfd29a2cJason Sams mSamplers[ct]->setupGL(); 6739c8bc7be5751ec52693d21abdf139c4dfd29a2cJason Sams } else { 686678e9b2568ad041429a2477177133fe4932159fJason Sams glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); 696678e9b2568ad041429a2477177133fe4932159fJason Sams glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); 706678e9b2568ad041429a2477177133fe4932159fJason Sams glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); 716678e9b2568ad041429a2477177133fe4932159fJason Sams glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT); 7239c8bc7be5751ec52693d21abdf139c4dfd29a2cJason Sams } 73326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 74326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams glActiveTexture(GL_TEXTURE0); 75326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 76326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 77326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 78326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid ProgramFragment::bindTexture(uint32_t slot, Allocation *a) 79326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 80326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (slot >= MAX_TEXTURE) { 81326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams LOGE("Attempt to bind a texture to a slot > MAX_TEXTURE"); 82326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams return; 83326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 84326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 85326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mTextures[slot].set(a); 86326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 87326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 88326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid ProgramFragment::bindSampler(uint32_t slot, Sampler *s) 89326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 90326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (slot >= MAX_TEXTURE) { 91326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams LOGE("Attempt to bind a Sampler to a slot > MAX_TEXTURE"); 92326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams return; 93326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 94326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 95326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mSamplers[slot].set(s); 96326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 97326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 98326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid ProgramFragment::setType(uint32_t slot, const Element *e, uint32_t dim) 99326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 100326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (slot >= MAX_TEXTURE) { 101326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams LOGE("Attempt to setType to a slot > MAX_TEXTURE"); 102326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams return; 103326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 104326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 105326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (dim >= 4) { 106326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams LOGE("Attempt to setType to a dimension > 3"); 107326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams return; 108326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 109326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 110326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mTextureFormats[slot].set(e); 111326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mTextureDimensions[slot] = dim; 112326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 113326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 114326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid ProgramFragment::setEnvMode(uint32_t slot, RsTexEnvMode env) 115326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 116326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (slot >= MAX_TEXTURE) { 117326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams LOGE("Attempt to setEnvMode to a slot > MAX_TEXTURE"); 118326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams return; 119326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 120326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 121326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mEnvModes[slot] = env; 122326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 123326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 124326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid ProgramFragment::setTexEnable(uint32_t slot, bool enable) 125326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 126326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (slot >= MAX_TEXTURE) { 127326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams LOGE("Attempt to setEnvMode to a slot > MAX_TEXTURE"); 128326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams return; 129326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 130326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 131326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams uint32_t bit = 1 << slot; 132326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mTextureEnableMask &= ~bit; 133326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (enable) { 134326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mTextureEnableMask |= bit; 135326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 136326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 137326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 138326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 139326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 140326e0ddf89e8df2837752fbfd7a014814b32082cJason SamsProgramFragmentState::ProgramFragmentState() 141326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 142326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams mPF = NULL; 143326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 144326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 145326e0ddf89e8df2837752fbfd7a014814b32082cJason SamsProgramFragmentState::~ProgramFragmentState() 146326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 147326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams delete mPF; 148326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 149326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 150326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 1518ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Samsvoid ProgramFragmentState::init(Context *rsc, int32_t w, int32_t h) 1528ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams{ 1538ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams ProgramFragment *pf = new ProgramFragment(NULL, NULL); 1548ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams mDefault.set(pf); 1558ce125be69531dbf3a7e856d5e59d1b8e2789db0Jason Sams} 156326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 157326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 158326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsnamespace android { 159326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsnamespace renderscript { 160326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 161326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid rsi_ProgramFragmentBegin(Context * rsc, RsElement in, RsElement out) 162326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 163326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams delete rsc->mStateFragment.mPF; 164326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams rsc->mStateFragment.mPF = new ProgramFragment((Element *)in, (Element *)out); 165326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 166326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 167326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid rsi_ProgramFragmentBindTexture(Context *rsc, RsProgramFragment vpf, uint32_t slot, RsAllocation a) 168326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 169326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ProgramFragment *pf = static_cast<ProgramFragment *>(vpf); 170326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams pf->bindTexture(slot, static_cast<Allocation *>(a)); 171326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 172326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams //LOGE("%p %p", pf, rsc->getFragment()); 173326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (pf == rsc->getFragment()) { 174326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams pf->setupGL(); 175326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 176326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 177326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 178326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid rsi_ProgramFragmentBindSampler(Context *rsc, RsProgramFragment vpf, uint32_t slot, RsSampler s) 179326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 180326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ProgramFragment *pf = static_cast<ProgramFragment *>(vpf); 181326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams pf->bindSampler(slot, static_cast<Sampler *>(s)); 182326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 183326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (pf == rsc->getFragment()) { 184326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams pf->setupGL(); 185326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 186326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 187326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 188326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid rsi_ProgramFragmentSetType(Context *rsc, uint32_t slot, RsType vt) 189326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 190326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams const Type *t = static_cast<const Type *>(vt); 191326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams uint32_t dim = 1; 192326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (t->getDimY()) { 193326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams dim ++; 194326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams if (t->getDimZ()) { 195326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams dim ++; 196326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 197326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams } 198326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 199326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams rsc->mStateFragment.mPF->setType(slot, t->getElement(), dim); 200326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 201326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 202326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid rsi_ProgramFragmentSetEnvMode(Context *rsc, uint32_t slot, RsTexEnvMode env) 203326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 204326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams rsc->mStateFragment.mPF->setEnvMode(slot, env); 205326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 206326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 207326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsvoid rsi_ProgramFragmentSetTexEnable(Context *rsc, uint32_t slot, bool enable) 208326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 209326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams rsc->mStateFragment.mPF->setTexEnable(slot, enable); 210326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 211326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 212326e0ddf89e8df2837752fbfd7a014814b32082cJason SamsRsProgramFragment rsi_ProgramFragmentCreate(Context *rsc) 213326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams{ 214326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams ProgramFragment *pf = rsc->mStateFragment.mPF; 215326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams pf->incRef(); 216326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams rsc->mStateFragment.mPF = 0; 217326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams return pf; 218326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 219326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 220a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Samsvoid rsi_ProgramFragmentDestroy(Context *rsc, RsProgramFragment vpf) 221a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams{ 222a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams ProgramFragment *pf = (ProgramFragment *)vpf; 223a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams if (pf->getName()) { 224a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams rsc->removeName(pf); 225a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams } 226a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams pf->decRef(); 227a0a1b6fbece2eb8d72d788422ab3e5f58d5a9216Jason Sams} 228326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 229326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 230326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 231326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams} 232326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams 233