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