blob: 8230cbcf4009611f385f21401c8dfb104f2c1eba [file] [log] [blame]
Jason Samsd19f10d2009-05-22 14:03:28 -07001/*
2 * Copyright (C) 2009 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "rsContext.h"
18#include "rsScriptC.h"
19#include "rsMatrix.h"
20
Jack Palevichec5a20b2009-05-28 15:53:04 -070021#include "acc/acc.h"
Joe Onorato3370ec92009-08-09 11:39:02 -070022#include "utils/Timers.h"
Jack Palevichec5a20b2009-05-28 15:53:04 -070023
Jason Sams4b962e52009-06-22 17:15:15 -070024#include <GLES/gl.h>
25#include <GLES/glext.h>
26
Jason Samsd19f10d2009-05-22 14:03:28 -070027using namespace android;
28using namespace android::renderscript;
29
Jason Sams462d11b2009-06-19 16:03:18 -070030#define GET_TLS() Context::ScriptTLSStruct * tls = \
31 (Context::ScriptTLSStruct *)pthread_getspecific(Context::gThreadTLSKey); \
32 Context * rsc = tls->mContext; \
33 ScriptC * sc = (ScriptC *) tls->mScript
34
Jason Samsd19f10d2009-05-22 14:03:28 -070035
36ScriptC::ScriptC()
37{
Jack Palevichec5a20b2009-05-28 15:53:04 -070038 mAccScript = NULL;
Jason Sams3a833d82009-06-08 15:20:31 -070039 memset(&mProgram, 0, sizeof(mProgram));
Jason Samsd19f10d2009-05-22 14:03:28 -070040}
41
42ScriptC::~ScriptC()
43{
Jack Palevichec5a20b2009-05-28 15:53:04 -070044 if (mAccScript) {
45 accDeleteScript(mAccScript);
46 }
Jason Samsd19f10d2009-05-22 14:03:28 -070047}
48
Jason Samsd19f10d2009-05-22 14:03:28 -070049
Jason Sams462d11b2009-06-19 16:03:18 -070050bool ScriptC::run(Context *rsc, uint32_t launchIndex)
Jason Samsd19f10d2009-05-22 14:03:28 -070051{
Jason Sams22534172009-08-04 16:58:20 -070052 Context::ScriptTLSStruct * tls =
Jason Sams462d11b2009-06-19 16:03:18 -070053 (Context::ScriptTLSStruct *)pthread_getspecific(Context::gThreadTLSKey);
Jason Sams3eaa3382009-06-10 15:04:38 -070054
55 if (mEnviroment.mFragmentStore.get()) {
56 rsc->setFragmentStore(mEnviroment.mFragmentStore.get());
57 }
58 if (mEnviroment.mFragment.get()) {
59 rsc->setFragment(mEnviroment.mFragment.get());
60 }
Jason Sams9c54bdb2009-06-17 16:52:59 -070061 if (mEnviroment.mVertex.get()) {
62 rsc->setVertex(mEnviroment.mVertex.get());
63 }
Jason Sams3eaa3382009-06-10 15:04:38 -070064
Joe Onorato3370ec92009-08-09 11:39:02 -070065 if (launchIndex == 0) {
66 mEnviroment.mStartTimeMillis
67 = nanoseconds_to_milliseconds(systemTime(SYSTEM_TIME_MONOTONIC));
68 }
69
Jason Sams2525a812009-09-03 15:43:13 -070070 for (int ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
71 if (mProgram.mSlotPointers[ct]) {
72 *mProgram.mSlotPointers[ct] = mSlots[ct]->getPtr();
73 }
74 }
75
Jason Samsc97bb882009-07-20 14:31:06 -070076 bool ret = false;
Jason Sams462d11b2009-06-19 16:03:18 -070077 tls->mScript = this;
Jason Samsc97bb882009-07-20 14:31:06 -070078 ret = mProgram.mScript(launchIndex) != 0;
Jason Sams462d11b2009-06-19 16:03:18 -070079 tls->mScript = NULL;
Jason Samsc97bb882009-07-20 14:31:06 -070080 return ret;
Jason Samsd19f10d2009-05-22 14:03:28 -070081}
82
83ScriptCState::ScriptCState()
84{
85 clear();
86}
87
88ScriptCState::~ScriptCState()
89{
Jack Palevichec5a20b2009-05-28 15:53:04 -070090 if (mAccScript) {
91 accDeleteScript(mAccScript);
92 }
Jason Samsd19f10d2009-05-22 14:03:28 -070093}
94
95void ScriptCState::clear()
96{
Jason Sams3a833d82009-06-08 15:20:31 -070097 memset(&mProgram, 0, sizeof(mProgram));
98
Jason Sams43ee06852009-08-12 17:54:11 -070099 for (uint32_t ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
100 mConstantBufferTypes[ct].clear();
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700101 mSlotNames[ct].setTo("");
Jason Sams334ea0c2009-08-17 13:56:09 -0700102 mSlotWritable[ct] = false;
Jason Sams43ee06852009-08-12 17:54:11 -0700103 }
Jason Sams3a833d82009-06-08 15:20:31 -0700104
105 memset(&mEnviroment, 0, sizeof(mEnviroment));
106 mEnviroment.mClearColor[0] = 0;
107 mEnviroment.mClearColor[1] = 0;
108 mEnviroment.mClearColor[2] = 0;
109 mEnviroment.mClearColor[3] = 1;
110 mEnviroment.mClearDepth = 1;
111 mEnviroment.mClearStencil = 0;
112 mEnviroment.mIsRoot = false;
Jason Sams3a833d82009-06-08 15:20:31 -0700113
Jack Palevichec5a20b2009-05-28 15:53:04 -0700114 mAccScript = NULL;
Jason Sams3a833d82009-06-08 15:20:31 -0700115
Joe Onoratod7b37742009-08-09 22:57:44 -0700116 mInt32Defines.clear();
117 mFloatDefines.clear();
Jason Sams39ddc9502009-06-05 17:35:09 -0700118}
119
Jason Sams22534172009-08-04 16:58:20 -0700120static ACCvoid* symbolLookup(ACCvoid* pContext, const ACCchar* name)
Jason Sams54440a02009-07-16 15:08:06 -0700121{
122 const ScriptCState::SymbolTable_t *sym = ScriptCState::lookupSymbol(name);
Jason Sams54440a02009-07-16 15:08:06 -0700123 if (sym) {
124 return sym->mPtr;
125 }
Jason Sams54440a02009-07-16 15:08:06 -0700126 LOGE("ScriptC sym lookup failed for %s", name);
Jason Sams54440a02009-07-16 15:08:06 -0700127 return NULL;
128}
Jason Samsd5680f92009-06-10 18:39:40 -0700129
Jason Sams3eaa3382009-06-10 15:04:38 -0700130void ScriptCState::runCompiler(Context *rsc)
Jason Sams39ddc9502009-06-05 17:35:09 -0700131{
132 mAccScript = accCreateScript();
Jason Samsd5680f92009-06-10 18:39:40 -0700133 String8 tmp;
Jason Sams39ddc9502009-06-05 17:35:09 -0700134
Jason Samsd5680f92009-06-10 18:39:40 -0700135 rsc->appendNameDefines(&tmp);
Jason Sams764205c2009-07-16 17:47:40 -0700136 appendDecls(&tmp);
Joe Onoratod7b37742009-08-09 22:57:44 -0700137 rsc->appendVarDefines(&tmp);
138 appendVarDefines(&tmp);
Jason Sams43ee06852009-08-12 17:54:11 -0700139 appendTypes(&tmp);
Jason Samsee411122009-07-21 12:20:54 -0700140 tmp.append("#line 1\n");
Jason Samsd5680f92009-06-10 18:39:40 -0700141
142 const char* scriptSource[] = {tmp.string(), mProgram.mScriptText};
143 int scriptLength[] = {tmp.length(), mProgram.mScriptTextLength} ;
144 accScriptSource(mAccScript, sizeof(scriptLength) / sizeof(int), scriptSource, scriptLength);
Jason Sams54440a02009-07-16 15:08:06 -0700145 accRegisterSymbolCallback(mAccScript, symbolLookup, NULL);
Jason Sams39ddc9502009-06-05 17:35:09 -0700146 accCompileScript(mAccScript);
Jason Sams3a833d82009-06-08 15:20:31 -0700147 accGetScriptLabel(mAccScript, "main", (ACCvoid**) &mProgram.mScript);
Jason Sams2525a812009-09-03 15:43:13 -0700148 accGetScriptLabel(mAccScript, "init", (ACCvoid**) &mProgram.mInit);
Jason Samsda423d82009-06-09 12:15:30 -0700149 rsAssert(mProgram.mScript);
150
Jason Sams764205c2009-07-16 17:47:40 -0700151 if (!mProgram.mScript) {
152 ACCchar buf[4096];
153 ACCsizei len;
154 accGetScriptInfoLog(mAccScript, sizeof(buf), &len, buf);
155 LOGE(buf);
Jason Sams764205c2009-07-16 17:47:40 -0700156 }
157
Jason Sams2525a812009-09-03 15:43:13 -0700158 if (mProgram.mInit) {
159 mProgram.mInit();
160 }
161
162 for (int ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
163 if (mSlotNames[ct].length() > 0) {
164 accGetScriptLabel(mAccScript,
165 mSlotNames[ct].string(),
166 (ACCvoid**) &mProgram.mSlotPointers[ct]);
167 LOGE("var %s %p", mSlotNames[ct].string(), mProgram.mSlotPointers[ct]);
168 }
169 }
170
Jason Sams9c54bdb2009-06-17 16:52:59 -0700171 mEnviroment.mFragment.set(rsc->getDefaultProgramFragment());
172 mEnviroment.mVertex.set(rsc->getDefaultProgramVertex());
173 mEnviroment.mFragmentStore.set(rsc->getDefaultProgramFragmentStore());
Jason Samsd5680f92009-06-10 18:39:40 -0700174
Jason Samsda423d82009-06-09 12:15:30 -0700175 if (mProgram.mScript) {
176 const static int pragmaMax = 16;
177 ACCsizei pragmaCount;
178 ACCchar * str[pragmaMax];
179 accGetPragmas(mAccScript, &pragmaCount, pragmaMax, &str[0]);
180
Jason Samsda423d82009-06-09 12:15:30 -0700181 for (int ct=0; ct < pragmaCount; ct+=2) {
Jason Samsda423d82009-06-09 12:15:30 -0700182 if (!strcmp(str[ct], "version")) {
183 continue;
Jason Samsda423d82009-06-09 12:15:30 -0700184 }
185
Jason Samsda423d82009-06-09 12:15:30 -0700186 if (!strcmp(str[ct], "stateVertex")) {
Jason Sams9c54bdb2009-06-17 16:52:59 -0700187 if (!strcmp(str[ct+1], "default")) {
188 continue;
189 }
190 if (!strcmp(str[ct+1], "parent")) {
191 mEnviroment.mVertex.clear();
192 continue;
193 }
194 ProgramVertex * pv = (ProgramVertex *)rsc->lookupName(str[ct+1]);
195 if (pv != NULL) {
196 mEnviroment.mVertex.set(pv);
197 continue;
198 }
Jason Samsda423d82009-06-09 12:15:30 -0700199 LOGE("Unreconized value %s passed to stateVertex", str[ct+1]);
200 }
201
202 if (!strcmp(str[ct], "stateRaster")) {
Jason Samsda423d82009-06-09 12:15:30 -0700203 LOGE("Unreconized value %s passed to stateRaster", str[ct+1]);
204 }
205
206 if (!strcmp(str[ct], "stateFragment")) {
Jason Sams9c54bdb2009-06-17 16:52:59 -0700207 if (!strcmp(str[ct+1], "default")) {
208 continue;
209 }
210 if (!strcmp(str[ct+1], "parent")) {
211 mEnviroment.mFragment.clear();
212 continue;
213 }
214 ProgramFragment * pf = (ProgramFragment *)rsc->lookupName(str[ct+1]);
Jason Sams3eaa3382009-06-10 15:04:38 -0700215 if (pf != NULL) {
216 mEnviroment.mFragment.set(pf);
Jason Samsda423d82009-06-09 12:15:30 -0700217 continue;
218 }
219 LOGE("Unreconized value %s passed to stateFragment", str[ct+1]);
220 }
221
222 if (!strcmp(str[ct], "stateFragmentStore")) {
Jason Sams9c54bdb2009-06-17 16:52:59 -0700223 if (!strcmp(str[ct+1], "default")) {
224 continue;
225 }
226 if (!strcmp(str[ct+1], "parent")) {
227 mEnviroment.mFragmentStore.clear();
228 continue;
229 }
Jason Sams22534172009-08-04 16:58:20 -0700230 ProgramFragmentStore * pfs =
Jason Sams3eaa3382009-06-10 15:04:38 -0700231 (ProgramFragmentStore *)rsc->lookupName(str[ct+1]);
232 if (pfs != NULL) {
233 mEnviroment.mFragmentStore.set(pfs);
Jason Samsda423d82009-06-09 12:15:30 -0700234 continue;
235 }
Jason Samsda423d82009-06-09 12:15:30 -0700236 LOGE("Unreconized value %s passed to stateFragmentStore", str[ct+1]);
237 }
238
239 }
240
Jason Sams22534172009-08-04 16:58:20 -0700241
Jason Samsda423d82009-06-09 12:15:30 -0700242 } else {
243 // Deal with an error.
244 }
Jason Samsd19f10d2009-05-22 14:03:28 -0700245}
246
Jason Sams2525a812009-09-03 15:43:13 -0700247static void appendElementBody(String8 *s, const Element *e)
248{
249 s->append(" {\n");
250 for (size_t ct2=0; ct2 < e->getComponentCount(); ct2++) {
251 const Component *c = e->getComponent(ct2);
252 s->append(" ");
253 s->append(c->getCType());
254 s->append(" ");
255 s->append(c->getComponentName());
256 s->append(";\n");
257 }
258 s->append("}");
259}
Joe Onoratod7b37742009-08-09 22:57:44 -0700260
261void ScriptCState::appendVarDefines(String8 *str)
262{
263 char buf[256];
264 LOGD("appendVarDefines mInt32Defines.size()=%d mFloatDefines.size()=%d\n",
265 mInt32Defines.size(), mFloatDefines.size());
266 for (size_t ct=0; ct < mInt32Defines.size(); ct++) {
267 str->append("#define ");
268 str->append(mInt32Defines.keyAt(ct));
269 str->append(" ");
270 sprintf(buf, "%i\n", (int)mInt32Defines.valueAt(ct));
271 str->append(buf);
272 }
273 for (size_t ct=0; ct < mFloatDefines.size(); ct++) {
274 str->append("#define ");
275 str->append(mFloatDefines.keyAt(ct));
276 str->append(" ");
277 sprintf(buf, "%ff\n", mFloatDefines.valueAt(ct));
278 str->append(buf);
279 }
280}
281
Jason Sams2525a812009-09-03 15:43:13 -0700282
283
Jason Sams43ee06852009-08-12 17:54:11 -0700284void ScriptCState::appendTypes(String8 *str)
285{
286 char buf[256];
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700287 String8 tmp;
Jason Sams43ee06852009-08-12 17:54:11 -0700288
Jason Samsea84a7c2009-09-04 14:42:41 -0700289 str->append("struct vec2_s {float x; float y;};");
290 str->append("struct vec3_s {float x; float y; float z;};");
291 str->append("struct vec4_s {float x; float y; float z; float w;};");
292
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700293 for (size_t ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
Jason Sams43ee06852009-08-12 17:54:11 -0700294 const Type *t = mConstantBufferTypes[ct].get();
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700295 if (!t) {
Jason Sams43ee06852009-08-12 17:54:11 -0700296 continue;
297 }
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700298 const Element *e = t->getElement();
Jason Sams2525a812009-09-03 15:43:13 -0700299 if (e->getName() && (e->getComponentCount() > 1)) {
300 String8 s("struct struct_");
301 s.append(e->getName());
302 appendElementBody(&s, e);
303 s.append(";\n");
304 s.append("#define ");
305 s.append(e->getName());
306 s.append("_t struct struct_");
307 s.append(e->getName());
308 s.append("\n\n");
309 LOGD(s);
310 str->append(s);
311 }
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700312
313 if (t->getName()) {
314 for (size_t ct2=0; ct2 < e->getComponentCount(); ct2++) {
315 const Component *c = e->getComponent(ct2);
316 tmp.setTo("#define OFFSETOF_");
317 tmp.append(t->getName());
318 tmp.append("_");
319 tmp.append(c->getComponentName());
320 sprintf(buf, " %i\n", ct2);
321 tmp.append(buf);
Jason Sams2525a812009-09-03 15:43:13 -0700322 LOGD(tmp);
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700323 str->append(tmp);
324 }
Jason Sams43ee06852009-08-12 17:54:11 -0700325 }
326
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700327 if (mSlotNames[ct].length() > 0) {
Jason Sams2525a812009-09-03 15:43:13 -0700328 String8 s;
329 if (e->getComponentCount() > 1) {
330 if (e->getName()) {
331 // Use the named struct
332 s.setTo(e->getName());
333 s.append("_t *");
334 } else {
335 // create an struct named from the slot.
336 s.setTo("struct ");
337 s.append(mSlotNames[ct]);
338 s.append("_s");
339 appendElementBody(&s, e);
340 s.append(";\n");
341 s.append("struct ");
342 s.append(mSlotNames[ct]);
343 s.append("_s * ");
344 }
345 } else {
346 // Just make an array
347 s.setTo(e->getComponent(0)->getCType());
348 s.append("_t *");
349 }
350 s.append(mSlotNames[ct]);
351 s.append(";\n");
352 LOGD(s);
353 str->append(s);
354#if 0
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700355 for (size_t ct2=0; ct2 < e->getComponentCount(); ct2++) {
356 const Component *c = e->getComponent(ct2);
357 tmp.setTo("#define ");
358 tmp.append(mSlotNames[ct]);
359 tmp.append("_");
360 tmp.append(c->getComponentName());
361 switch (c->getType()) {
362 case Component::FLOAT:
363 tmp.append(" loadF(");
364 break;
365 case Component::SIGNED:
366 sprintf(buf, " loadI%i(", c->getBits());
367 tmp.append(buf);
368 break;
369 case Component::UNSIGNED:
370 sprintf(buf, " loadU%i(", c->getBits());
371 tmp.append(buf);
372 break;
373 }
374 sprintf(buf, "%i, %i)\n", ct, ct2);
375 tmp.append(buf);
376
Jason Sams2525a812009-09-03 15:43:13 -0700377 LOGD(tmp);
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700378 str->append(tmp);
379 }
Jason Sams2525a812009-09-03 15:43:13 -0700380#endif
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700381 }
Jason Sams43ee06852009-08-12 17:54:11 -0700382 }
Jason Sams43ee06852009-08-12 17:54:11 -0700383}
384
Joe Onoratod7b37742009-08-09 22:57:44 -0700385
Jason Samsd19f10d2009-05-22 14:03:28 -0700386namespace android {
387namespace renderscript {
388
389void rsi_ScriptCBegin(Context * rsc)
390{
391 ScriptCState *ss = &rsc->mScriptC;
392 ss->clear();
393}
394
Jason Sams3a833d82009-06-08 15:20:31 -0700395void rsi_ScriptCSetScript(Context * rsc, void *vp)
Jason Samsd19f10d2009-05-22 14:03:28 -0700396{
397 ScriptCState *ss = &rsc->mScriptC;
Jason Samsc97bb882009-07-20 14:31:06 -0700398 ss->mProgram.mScript = reinterpret_cast<ScriptC::RunScript_t>(vp);
Jason Samsd19f10d2009-05-22 14:03:28 -0700399}
400
Jason Sams39ddc9502009-06-05 17:35:09 -0700401void rsi_ScriptCSetText(Context *rsc, const char *text, uint32_t len)
402{
403 ScriptCState *ss = &rsc->mScriptC;
Jason Sams3a833d82009-06-08 15:20:31 -0700404 ss->mProgram.mScriptText = text;
405 ss->mProgram.mScriptTextLength = len;
Jason Sams39ddc9502009-06-05 17:35:09 -0700406}
407
408
Jason Samsd19f10d2009-05-22 14:03:28 -0700409RsScript rsi_ScriptCCreate(Context * rsc)
410{
411 ScriptCState *ss = &rsc->mScriptC;
412
Jason Sams3eaa3382009-06-10 15:04:38 -0700413 ss->runCompiler(rsc);
Jason Sams39ddc9502009-06-05 17:35:09 -0700414
Jason Samsd19f10d2009-05-22 14:03:28 -0700415 ScriptC *s = new ScriptC();
Jason Sams07ae4062009-08-27 20:23:34 -0700416 s->incUserRef();
Jack Palevichec5a20b2009-05-28 15:53:04 -0700417 s->mAccScript = ss->mAccScript;
418 ss->mAccScript = NULL;
Jason Sams3a833d82009-06-08 15:20:31 -0700419 s->mEnviroment = ss->mEnviroment;
420 s->mProgram = ss->mProgram;
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700421 for (int ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
422 s->mTypes[ct].set(ss->mConstantBufferTypes[ct].get());
423 s->mSlotNames[ct] = ss->mSlotNames[ct];
Jason Sams334ea0c2009-08-17 13:56:09 -0700424 s->mSlotWritable[ct] = ss->mSlotWritable[ct];
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700425 }
Jason Samsda423d82009-06-09 12:15:30 -0700426
Jason Samsfbf0b9e2009-08-13 12:59:04 -0700427 ss->clear();
Jason Samsd19f10d2009-05-22 14:03:28 -0700428 return s;
429}
430
Joe Onoratod7b37742009-08-09 22:57:44 -0700431void rsi_ScriptCSetDefineF(Context *rsc, const char* name, float value)
432{
433 ScriptCState *ss = &rsc->mScriptC;
434 ss->mFloatDefines.add(String8(name), value);
435}
436
437void rsi_ScriptCSetDefineI32(Context *rsc, const char* name, int32_t value)
438{
439 ScriptCState *ss = &rsc->mScriptC;
440 ss->mInt32Defines.add(String8(name), value);
441}
442
Jason Samsd19f10d2009-05-22 14:03:28 -0700443}
444}
445
446