blob: bff337c0b90d274cf7f4c63228621cb0b5973ae5 [file] [log] [blame]
Jason Sams326e0dd2009-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 Palevich1ef8b802009-05-28 15:53:04 -070021#include "acc/acc.h"
Joe Onorato9c4e4ca2009-08-09 11:39:02 -070022#include "utils/Timers.h"
Jack Palevich1ef8b802009-05-28 15:53:04 -070023
Jason Sams1aa5a4e2009-06-22 17:15:15 -070024#include <GLES/gl.h>
25#include <GLES/glext.h>
26
Jason Sams326e0dd2009-05-22 14:03:28 -070027using namespace android;
28using namespace android::renderscript;
29
Jason Samse5769102009-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 Sams326e0dd2009-05-22 14:03:28 -070035
36ScriptC::ScriptC()
37{
Jack Palevich1ef8b802009-05-28 15:53:04 -070038 mAccScript = NULL;
Jason Samsefb8de12009-06-08 15:20:31 -070039 memset(&mProgram, 0, sizeof(mProgram));
Jason Sams326e0dd2009-05-22 14:03:28 -070040}
41
42ScriptC::~ScriptC()
43{
Jack Palevich1ef8b802009-05-28 15:53:04 -070044 if (mAccScript) {
45 accDeleteScript(mAccScript);
46 }
Jason Sams326e0dd2009-05-22 14:03:28 -070047}
48
Jason Samsada7f272009-09-24 14:55:38 -070049void ScriptC::setupScript()
50{
51 for (int ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
52 if (mProgram.mSlotPointers[ct]) {
53 *mProgram.mSlotPointers[ct] = mSlots[ct]->getPtr();
54 }
55 }
56}
57
Jason Sams326e0dd2009-05-22 14:03:28 -070058
Jason Samse5769102009-06-19 16:03:18 -070059bool ScriptC::run(Context *rsc, uint32_t launchIndex)
Jason Sams326e0dd2009-05-22 14:03:28 -070060{
Jason Samsd34b7252009-08-04 16:58:20 -070061 Context::ScriptTLSStruct * tls =
Jason Samse5769102009-06-19 16:03:18 -070062 (Context::ScriptTLSStruct *)pthread_getspecific(Context::gThreadTLSKey);
Jason Samsa0a1b6f2009-06-10 15:04:38 -070063
64 if (mEnviroment.mFragmentStore.get()) {
65 rsc->setFragmentStore(mEnviroment.mFragmentStore.get());
66 }
67 if (mEnviroment.mFragment.get()) {
68 rsc->setFragment(mEnviroment.mFragment.get());
69 }
Jason Sams8ce125b2009-06-17 16:52:59 -070070 if (mEnviroment.mVertex.get()) {
71 rsc->setVertex(mEnviroment.mVertex.get());
72 }
Jason Samsa0a1b6f2009-06-10 15:04:38 -070073
Joe Onorato9c4e4ca2009-08-09 11:39:02 -070074 if (launchIndex == 0) {
75 mEnviroment.mStartTimeMillis
76 = nanoseconds_to_milliseconds(systemTime(SYSTEM_TIME_MONOTONIC));
77 }
Jason Samsada7f272009-09-24 14:55:38 -070078 setupScript();
Jason Sams1d54f102009-09-03 15:43:13 -070079
Jason Samse45ac6e2009-07-20 14:31:06 -070080 bool ret = false;
Jason Samse5769102009-06-19 16:03:18 -070081 tls->mScript = this;
Jason Samse45ac6e2009-07-20 14:31:06 -070082 ret = mProgram.mScript(launchIndex) != 0;
Jason Samse5769102009-06-19 16:03:18 -070083 tls->mScript = NULL;
Jason Samse45ac6e2009-07-20 14:31:06 -070084 return ret;
Jason Sams326e0dd2009-05-22 14:03:28 -070085}
86
87ScriptCState::ScriptCState()
88{
Jason Sams8c6bc692009-09-16 15:04:38 -070089 mScript = NULL;
Jason Sams326e0dd2009-05-22 14:03:28 -070090 clear();
91}
92
93ScriptCState::~ScriptCState()
94{
Jason Sams8c6bc692009-09-16 15:04:38 -070095 delete mScript;
96 mScript = NULL;
Jason Sams326e0dd2009-05-22 14:03:28 -070097}
98
99void ScriptCState::clear()
100{
Jason Sams8b2c0652009-08-12 17:54:11 -0700101 for (uint32_t ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
102 mConstantBufferTypes[ct].clear();
Jason Samsfa517192009-08-13 12:59:04 -0700103 mSlotNames[ct].setTo("");
Jason Sams8c6bc692009-09-16 15:04:38 -0700104 mInvokableNames[ct].setTo("");
Jason Sams90b36a82009-08-17 13:56:09 -0700105 mSlotWritable[ct] = false;
Jason Sams8b2c0652009-08-12 17:54:11 -0700106 }
Jason Samsefb8de12009-06-08 15:20:31 -0700107
Jason Sams8c6bc692009-09-16 15:04:38 -0700108 delete mScript;
109 mScript = new ScriptC();
Jason Samsefb8de12009-06-08 15:20:31 -0700110
Joe Onorato57b79ce2009-08-09 22:57:44 -0700111 mInt32Defines.clear();
112 mFloatDefines.clear();
Jason Sams1f526332009-06-05 17:35:09 -0700113}
114
Jason Samsd34b7252009-08-04 16:58:20 -0700115static ACCvoid* symbolLookup(ACCvoid* pContext, const ACCchar* name)
Jason Sams29df66f2009-07-16 15:08:06 -0700116{
117 const ScriptCState::SymbolTable_t *sym = ScriptCState::lookupSymbol(name);
Jason Sams29df66f2009-07-16 15:08:06 -0700118 if (sym) {
119 return sym->mPtr;
120 }
Jason Sams29df66f2009-07-16 15:08:06 -0700121 LOGE("ScriptC sym lookup failed for %s", name);
Jason Sams29df66f2009-07-16 15:08:06 -0700122 return NULL;
123}
Jason Samsa4a54e42009-06-10 18:39:40 -0700124
Jason Sams8c6bc692009-09-16 15:04:38 -0700125void ScriptCState::runCompiler(Context *rsc, ScriptC *s)
Jason Sams1f526332009-06-05 17:35:09 -0700126{
Jason Sams8c6bc692009-09-16 15:04:38 -0700127 s->mAccScript = accCreateScript();
Jason Samsa4a54e42009-06-10 18:39:40 -0700128 String8 tmp;
Jason Sams1f526332009-06-05 17:35:09 -0700129
Jason Samsa4a54e42009-06-10 18:39:40 -0700130 rsc->appendNameDefines(&tmp);
Jason Samsf1685042009-07-16 17:47:40 -0700131 appendDecls(&tmp);
Joe Onorato57b79ce2009-08-09 22:57:44 -0700132 rsc->appendVarDefines(&tmp);
133 appendVarDefines(&tmp);
Jason Sams8b2c0652009-08-12 17:54:11 -0700134 appendTypes(&tmp);
Jason Samsb5909ce2009-07-21 12:20:54 -0700135 tmp.append("#line 1\n");
Jason Samsa4a54e42009-06-10 18:39:40 -0700136
Jason Sams8c6bc692009-09-16 15:04:38 -0700137 const char* scriptSource[] = {tmp.string(), s->mEnviroment.mScriptText};
138 int scriptLength[] = {tmp.length(), s->mEnviroment.mScriptTextLength} ;
139 accScriptSource(s->mAccScript, sizeof(scriptLength) / sizeof(int), scriptSource, scriptLength);
140 accRegisterSymbolCallback(s->mAccScript, symbolLookup, NULL);
141 accCompileScript(s->mAccScript);
142 accGetScriptLabel(s->mAccScript, "main", (ACCvoid**) &s->mProgram.mScript);
143 accGetScriptLabel(s->mAccScript, "init", (ACCvoid**) &s->mProgram.mInit);
144 rsAssert(s->mProgram.mScript);
Jason Sams10308932009-06-09 12:15:30 -0700145
Jason Sams8c6bc692009-09-16 15:04:38 -0700146 if (!s->mProgram.mScript) {
Jason Samsf1685042009-07-16 17:47:40 -0700147 ACCchar buf[4096];
148 ACCsizei len;
Jason Sams8c6bc692009-09-16 15:04:38 -0700149 accGetScriptInfoLog(s->mAccScript, sizeof(buf), &len, buf);
Jason Samsf1685042009-07-16 17:47:40 -0700150 LOGE(buf);
Jason Samsf1685042009-07-16 17:47:40 -0700151 }
152
Jason Sams8c6bc692009-09-16 15:04:38 -0700153 if (s->mProgram.mInit) {
154 s->mProgram.mInit();
Jason Sams1d54f102009-09-03 15:43:13 -0700155 }
156
157 for (int ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
158 if (mSlotNames[ct].length() > 0) {
Jason Sams8c6bc692009-09-16 15:04:38 -0700159 accGetScriptLabel(s->mAccScript,
Jason Sams1d54f102009-09-03 15:43:13 -0700160 mSlotNames[ct].string(),
Jason Sams8c6bc692009-09-16 15:04:38 -0700161 (ACCvoid**) &s->mProgram.mSlotPointers[ct]);
Jason Sams1d54f102009-09-03 15:43:13 -0700162 }
163 }
164
Jason Sams8c6bc692009-09-16 15:04:38 -0700165 for (int ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
166 if (mInvokableNames[ct].length() > 0) {
167 accGetScriptLabel(s->mAccScript,
168 mInvokableNames[ct].string(),
169 (ACCvoid**) &s->mEnviroment.mInvokables[ct]);
170 }
171 }
Jason Samsa4a54e42009-06-10 18:39:40 -0700172
Jason Sams8c6bc692009-09-16 15:04:38 -0700173 s->mEnviroment.mFragment.set(rsc->getDefaultProgramFragment());
174 s->mEnviroment.mVertex.set(rsc->getDefaultProgramVertex());
175 s->mEnviroment.mFragmentStore.set(rsc->getDefaultProgramFragmentStore());
176
177 if (s->mProgram.mScript) {
Jason Sams10308932009-06-09 12:15:30 -0700178 const static int pragmaMax = 16;
179 ACCsizei pragmaCount;
180 ACCchar * str[pragmaMax];
Jason Sams8c6bc692009-09-16 15:04:38 -0700181 accGetPragmas(s->mAccScript, &pragmaCount, pragmaMax, &str[0]);
Jason Sams10308932009-06-09 12:15:30 -0700182
Jason Sams10308932009-06-09 12:15:30 -0700183 for (int ct=0; ct < pragmaCount; ct+=2) {
Jason Sams10308932009-06-09 12:15:30 -0700184 if (!strcmp(str[ct], "version")) {
185 continue;
Jason Sams10308932009-06-09 12:15:30 -0700186 }
187
Jason Sams10308932009-06-09 12:15:30 -0700188 if (!strcmp(str[ct], "stateVertex")) {
Jason Sams8ce125b2009-06-17 16:52:59 -0700189 if (!strcmp(str[ct+1], "default")) {
190 continue;
191 }
192 if (!strcmp(str[ct+1], "parent")) {
Jason Sams8c6bc692009-09-16 15:04:38 -0700193 s->mEnviroment.mVertex.clear();
Jason Sams8ce125b2009-06-17 16:52:59 -0700194 continue;
195 }
196 ProgramVertex * pv = (ProgramVertex *)rsc->lookupName(str[ct+1]);
197 if (pv != NULL) {
Jason Sams8c6bc692009-09-16 15:04:38 -0700198 s->mEnviroment.mVertex.set(pv);
Jason Sams8ce125b2009-06-17 16:52:59 -0700199 continue;
200 }
Jason Sams10308932009-06-09 12:15:30 -0700201 LOGE("Unreconized value %s passed to stateVertex", str[ct+1]);
202 }
203
204 if (!strcmp(str[ct], "stateRaster")) {
Jason Sams10308932009-06-09 12:15:30 -0700205 LOGE("Unreconized value %s passed to stateRaster", str[ct+1]);
206 }
207
208 if (!strcmp(str[ct], "stateFragment")) {
Jason Sams8ce125b2009-06-17 16:52:59 -0700209 if (!strcmp(str[ct+1], "default")) {
210 continue;
211 }
212 if (!strcmp(str[ct+1], "parent")) {
Jason Sams8c6bc692009-09-16 15:04:38 -0700213 s->mEnviroment.mFragment.clear();
Jason Sams8ce125b2009-06-17 16:52:59 -0700214 continue;
215 }
216 ProgramFragment * pf = (ProgramFragment *)rsc->lookupName(str[ct+1]);
Jason Samsa0a1b6f2009-06-10 15:04:38 -0700217 if (pf != NULL) {
Jason Sams8c6bc692009-09-16 15:04:38 -0700218 s->mEnviroment.mFragment.set(pf);
Jason Sams10308932009-06-09 12:15:30 -0700219 continue;
220 }
221 LOGE("Unreconized value %s passed to stateFragment", str[ct+1]);
222 }
223
224 if (!strcmp(str[ct], "stateFragmentStore")) {
Jason Sams8ce125b2009-06-17 16:52:59 -0700225 if (!strcmp(str[ct+1], "default")) {
226 continue;
227 }
228 if (!strcmp(str[ct+1], "parent")) {
Jason Sams8c6bc692009-09-16 15:04:38 -0700229 s->mEnviroment.mFragmentStore.clear();
Jason Sams8ce125b2009-06-17 16:52:59 -0700230 continue;
231 }
Jason Samsd34b7252009-08-04 16:58:20 -0700232 ProgramFragmentStore * pfs =
Jason Samsa0a1b6f2009-06-10 15:04:38 -0700233 (ProgramFragmentStore *)rsc->lookupName(str[ct+1]);
234 if (pfs != NULL) {
Jason Sams8c6bc692009-09-16 15:04:38 -0700235 s->mEnviroment.mFragmentStore.set(pfs);
Jason Sams10308932009-06-09 12:15:30 -0700236 continue;
237 }
Jason Sams10308932009-06-09 12:15:30 -0700238 LOGE("Unreconized value %s passed to stateFragmentStore", str[ct+1]);
239 }
240
241 }
242
Jason Samsd34b7252009-08-04 16:58:20 -0700243
Jason Sams10308932009-06-09 12:15:30 -0700244 } else {
245 // Deal with an error.
246 }
Jason Sams326e0dd2009-05-22 14:03:28 -0700247}
248
Jason Sams1d54f102009-09-03 15:43:13 -0700249static void appendElementBody(String8 *s, const Element *e)
250{
251 s->append(" {\n");
252 for (size_t ct2=0; ct2 < e->getComponentCount(); ct2++) {
253 const Component *c = e->getComponent(ct2);
254 s->append(" ");
255 s->append(c->getCType());
256 s->append(" ");
257 s->append(c->getComponentName());
258 s->append(";\n");
259 }
260 s->append("}");
261}
Joe Onorato57b79ce2009-08-09 22:57:44 -0700262
263void ScriptCState::appendVarDefines(String8 *str)
264{
265 char buf[256];
266 LOGD("appendVarDefines mInt32Defines.size()=%d mFloatDefines.size()=%d\n",
267 mInt32Defines.size(), mFloatDefines.size());
268 for (size_t ct=0; ct < mInt32Defines.size(); ct++) {
269 str->append("#define ");
270 str->append(mInt32Defines.keyAt(ct));
271 str->append(" ");
272 sprintf(buf, "%i\n", (int)mInt32Defines.valueAt(ct));
273 str->append(buf);
274 }
275 for (size_t ct=0; ct < mFloatDefines.size(); ct++) {
276 str->append("#define ");
277 str->append(mFloatDefines.keyAt(ct));
278 str->append(" ");
279 sprintf(buf, "%ff\n", mFloatDefines.valueAt(ct));
280 str->append(buf);
281 }
282}
283
Jason Sams1d54f102009-09-03 15:43:13 -0700284
285
Jason Sams8b2c0652009-08-12 17:54:11 -0700286void ScriptCState::appendTypes(String8 *str)
287{
288 char buf[256];
Jason Samsfa517192009-08-13 12:59:04 -0700289 String8 tmp;
Jason Sams8b2c0652009-08-12 17:54:11 -0700290
Jason Samsa57c0a72009-09-04 14:42:41 -0700291 str->append("struct vec2_s {float x; float y;};");
292 str->append("struct vec3_s {float x; float y; float z;};");
293 str->append("struct vec4_s {float x; float y; float z; float w;};");
294
Jason Samsfa517192009-08-13 12:59:04 -0700295 for (size_t ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
Jason Sams8b2c0652009-08-12 17:54:11 -0700296 const Type *t = mConstantBufferTypes[ct].get();
Jason Samsfa517192009-08-13 12:59:04 -0700297 if (!t) {
Jason Sams8b2c0652009-08-12 17:54:11 -0700298 continue;
299 }
Jason Samsfa517192009-08-13 12:59:04 -0700300 const Element *e = t->getElement();
Jason Sams1d54f102009-09-03 15:43:13 -0700301 if (e->getName() && (e->getComponentCount() > 1)) {
302 String8 s("struct struct_");
303 s.append(e->getName());
304 appendElementBody(&s, e);
305 s.append(";\n");
306 s.append("#define ");
307 s.append(e->getName());
308 s.append("_t struct struct_");
309 s.append(e->getName());
310 s.append("\n\n");
311 LOGD(s);
312 str->append(s);
313 }
Jason Samsfa517192009-08-13 12:59:04 -0700314
315 if (t->getName()) {
316 for (size_t ct2=0; ct2 < e->getComponentCount(); ct2++) {
317 const Component *c = e->getComponent(ct2);
318 tmp.setTo("#define OFFSETOF_");
319 tmp.append(t->getName());
320 tmp.append("_");
321 tmp.append(c->getComponentName());
322 sprintf(buf, " %i\n", ct2);
323 tmp.append(buf);
Jason Sams1d54f102009-09-03 15:43:13 -0700324 LOGD(tmp);
Jason Samsfa517192009-08-13 12:59:04 -0700325 str->append(tmp);
326 }
Jason Sams8b2c0652009-08-12 17:54:11 -0700327 }
328
Jason Samsfa517192009-08-13 12:59:04 -0700329 if (mSlotNames[ct].length() > 0) {
Jason Sams1d54f102009-09-03 15:43:13 -0700330 String8 s;
331 if (e->getComponentCount() > 1) {
332 if (e->getName()) {
333 // Use the named struct
334 s.setTo(e->getName());
335 s.append("_t *");
336 } else {
337 // create an struct named from the slot.
338 s.setTo("struct ");
339 s.append(mSlotNames[ct]);
340 s.append("_s");
341 appendElementBody(&s, e);
342 s.append(";\n");
343 s.append("struct ");
344 s.append(mSlotNames[ct]);
345 s.append("_s * ");
346 }
347 } else {
348 // Just make an array
349 s.setTo(e->getComponent(0)->getCType());
350 s.append("_t *");
351 }
352 s.append(mSlotNames[ct]);
353 s.append(";\n");
354 LOGD(s);
355 str->append(s);
Jason Samsfa517192009-08-13 12:59:04 -0700356 }
Jason Sams8b2c0652009-08-12 17:54:11 -0700357 }
Jason Sams8b2c0652009-08-12 17:54:11 -0700358}
359
Joe Onorato57b79ce2009-08-09 22:57:44 -0700360
Jason Sams326e0dd2009-05-22 14:03:28 -0700361namespace android {
362namespace renderscript {
363
364void rsi_ScriptCBegin(Context * rsc)
365{
366 ScriptCState *ss = &rsc->mScriptC;
367 ss->clear();
368}
369
Jason Samsefb8de12009-06-08 15:20:31 -0700370void rsi_ScriptCSetScript(Context * rsc, void *vp)
Jason Sams326e0dd2009-05-22 14:03:28 -0700371{
Jason Sams8c6bc692009-09-16 15:04:38 -0700372 rsAssert(0);
373 //ScriptCState *ss = &rsc->mScriptC;
374 //ss->mProgram.mScript = reinterpret_cast<ScriptC::RunScript_t>(vp);
Jason Sams326e0dd2009-05-22 14:03:28 -0700375}
376
Jason Sams1f526332009-06-05 17:35:09 -0700377void rsi_ScriptCSetText(Context *rsc, const char *text, uint32_t len)
378{
379 ScriptCState *ss = &rsc->mScriptC;
Jason Sams8c6bc692009-09-16 15:04:38 -0700380 ss->mScript->mEnviroment.mScriptText = text;
381 ss->mScript->mEnviroment.mScriptTextLength = len;
Jason Sams1f526332009-06-05 17:35:09 -0700382}
383
384
Jason Sams326e0dd2009-05-22 14:03:28 -0700385RsScript rsi_ScriptCCreate(Context * rsc)
386{
387 ScriptCState *ss = &rsc->mScriptC;
388
Jason Sams8c6bc692009-09-16 15:04:38 -0700389 ScriptC *s = ss->mScript;
390 ss->mScript = NULL;
Jason Sams1f526332009-06-05 17:35:09 -0700391
Jason Sams8c6bc692009-09-16 15:04:38 -0700392 ss->runCompiler(rsc, s);
Jason Sams9397e302009-08-27 20:23:34 -0700393 s->incUserRef();
Jason Samsfa517192009-08-13 12:59:04 -0700394 for (int ct=0; ct < MAX_SCRIPT_BANKS; ct++) {
395 s->mTypes[ct].set(ss->mConstantBufferTypes[ct].get());
396 s->mSlotNames[ct] = ss->mSlotNames[ct];
Jason Sams90b36a82009-08-17 13:56:09 -0700397 s->mSlotWritable[ct] = ss->mSlotWritable[ct];
Jason Samsfa517192009-08-13 12:59:04 -0700398 }
Jason Sams10308932009-06-09 12:15:30 -0700399
Jason Samsfa517192009-08-13 12:59:04 -0700400 ss->clear();
Jason Sams326e0dd2009-05-22 14:03:28 -0700401 return s;
402}
403
Joe Onorato57b79ce2009-08-09 22:57:44 -0700404void rsi_ScriptCSetDefineF(Context *rsc, const char* name, float value)
405{
406 ScriptCState *ss = &rsc->mScriptC;
407 ss->mFloatDefines.add(String8(name), value);
408}
409
410void rsi_ScriptCSetDefineI32(Context *rsc, const char* name, int32_t value)
411{
412 ScriptCState *ss = &rsc->mScriptC;
413 ss->mInt32Defines.add(String8(name), value);
414}
415
Jason Sams326e0dd2009-05-22 14:03:28 -0700416}
417}
418
419