Zonr Chang | c383a50 | 2010-10-12 01:52:08 +0800 | [diff] [blame] | 1 | /* |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2 | * Copyright 2010-2014, The Android Open Source Project |
Zonr Chang | c383a50 | 2010-10-12 01:52:08 +0800 | [diff] [blame] | 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 | |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 17 | #include "slang_rs_reflection.h" |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 18 | |
Stephen Hines | e639eb5 | 2010-11-08 19:27:20 -0800 | [diff] [blame] | 19 | #include <sys/stat.h> |
| 20 | |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 21 | #include <cstdarg> |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 22 | #include <cctype> |
Stephen Hines | e639eb5 | 2010-11-08 19:27:20 -0800 | [diff] [blame] | 23 | |
| 24 | #include <algorithm> |
Stephen Hines | d369cda | 2012-02-13 12:00:03 -0800 | [diff] [blame] | 25 | #include <sstream> |
Stephen Hines | e639eb5 | 2010-11-08 19:27:20 -0800 | [diff] [blame] | 26 | #include <string> |
| 27 | #include <utility> |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 28 | |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 29 | #include "llvm/ADT/APFloat.h" |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 30 | #include "llvm/ADT/StringExtras.h" |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 31 | |
Raphael | 8d5a2f6 | 2011-02-08 00:15:05 -0800 | [diff] [blame] | 32 | #include "os_sep.h" |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 33 | #include "slang_rs_context.h" |
| 34 | #include "slang_rs_export_var.h" |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 35 | #include "slang_rs_export_foreach.h" |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 36 | #include "slang_rs_export_func.h" |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 37 | #include "slang_rs_export_reduce.h" |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 38 | #include "slang_rs_reflect_utils.h" |
Stephen Hines | 4cc499d | 2011-08-24 19:06:17 -0700 | [diff] [blame] | 39 | #include "slang_version.h" |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 40 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 41 | #define RS_SCRIPT_CLASS_NAME_PREFIX "ScriptC_" |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 42 | #define RS_SCRIPT_CLASS_SUPER_CLASS_NAME "ScriptC" |
Shih-wei Liao | 8b1d0dd | 2010-07-15 18:56:55 -0700 | [diff] [blame] | 43 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 44 | #define RS_TYPE_CLASS_SUPER_CLASS_NAME ".Script.FieldBase" |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 45 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 46 | #define RS_TYPE_ITEM_CLASS_NAME "Item" |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 47 | |
Tim Murray | 3a38b74 | 2014-07-02 10:41:08 -0700 | [diff] [blame] | 48 | #define RS_TYPE_ITEM_SIZEOF_LEGACY "Item.sizeof" |
| 49 | #define RS_TYPE_ITEM_SIZEOF_CURRENT "mElement.getBytesSize()" |
| 50 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 51 | #define RS_TYPE_ITEM_BUFFER_NAME "mItemArray" |
| 52 | #define RS_TYPE_ITEM_BUFFER_PACKER_NAME "mIOBuffer" |
| 53 | #define RS_TYPE_ELEMENT_REF_NAME "mElementCache" |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 54 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 55 | #define RS_EXPORT_VAR_INDEX_PREFIX "mExportVarIdx_" |
| 56 | #define RS_EXPORT_VAR_PREFIX "mExportVar_" |
| 57 | #define RS_EXPORT_VAR_ELEM_PREFIX "mExportVarElem_" |
| 58 | #define RS_EXPORT_VAR_DIM_PREFIX "mExportVarDim_" |
| 59 | #define RS_EXPORT_VAR_CONST_PREFIX "const_" |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 60 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 61 | #define RS_ELEM_PREFIX "__" |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 62 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 63 | #define RS_FP_PREFIX "__rs_fp_" |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 64 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 65 | #define RS_RESOURCE_NAME "__rs_resource_name" |
Stephen Hines | d293693 | 2012-09-12 20:44:32 -0700 | [diff] [blame] | 66 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 67 | #define RS_EXPORT_FUNC_INDEX_PREFIX "mExportFuncIdx_" |
| 68 | #define RS_EXPORT_FOREACH_INDEX_PREFIX "mExportForEachIdx_" |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 69 | #define RS_EXPORT_REDUCE_INDEX_PREFIX "mExportReduceIdx_" |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 70 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 71 | #define RS_EXPORT_VAR_ALLOCATION_PREFIX "mAlloction_" |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 72 | #define RS_EXPORT_VAR_DATA_STORAGE_PREFIX "mData_" |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 73 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 74 | #define SAVED_RS_REFERENCE "mRSLocal" |
| 75 | |
Stephen Hines | e639eb5 | 2010-11-08 19:27:20 -0800 | [diff] [blame] | 76 | namespace slang { |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 77 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 78 | class RSReflectionJavaElementBuilder { |
| 79 | public: |
| 80 | RSReflectionJavaElementBuilder(const char *ElementBuilderName, |
| 81 | const RSExportRecordType *ERT, |
| 82 | const char *RenderScriptVar, |
| 83 | GeneratedFile *Out, const RSContext *RSContext, |
| 84 | RSReflectionJava *Reflection); |
| 85 | void generate(); |
| 86 | |
| 87 | private: |
| 88 | void genAddElement(const RSExportType *ET, const std::string &VarName, |
| 89 | unsigned ArraySize); |
| 90 | void genAddStatementStart(); |
| 91 | void genAddStatementEnd(const std::string &VarName, unsigned ArraySize); |
| 92 | void genAddPadding(int PaddingSize); |
| 93 | // TODO Will remove later due to field name information is not necessary for |
| 94 | // C-reflect-to-Java |
| 95 | std::string createPaddingField() { |
| 96 | return mPaddingPrefix + llvm::itostr(mPaddingFieldIndex++); |
| 97 | } |
| 98 | |
| 99 | const char *mElementBuilderName; |
| 100 | const RSExportRecordType *mERT; |
| 101 | const char *mRenderScriptVar; |
| 102 | GeneratedFile *mOut; |
| 103 | std::string mPaddingPrefix; |
| 104 | int mPaddingFieldIndex; |
| 105 | const RSContext *mRSContext; |
| 106 | RSReflectionJava *mReflection; |
| 107 | }; |
| 108 | |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 109 | static const char *GetMatrixTypeName(const RSExportMatrixType *EMT) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 110 | static const char *MatrixTypeJavaNameMap[] = {/* 2x2 */ "Matrix2f", |
| 111 | /* 3x3 */ "Matrix3f", |
| 112 | /* 4x4 */ "Matrix4f", |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 113 | }; |
| 114 | unsigned Dim = EMT->getDim(); |
| 115 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 116 | if ((Dim - 2) < (sizeof(MatrixTypeJavaNameMap) / sizeof(const char *))) |
| 117 | return MatrixTypeJavaNameMap[EMT->getDim() - 2]; |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 118 | |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 119 | slangAssert(false && "GetMatrixTypeName : Unsupported matrix dimension"); |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 120 | return nullptr; |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 121 | } |
| 122 | |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 123 | static const char *GetVectorAccessor(unsigned Index) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 124 | static const char *VectorAccessorMap[] = {/* 0 */ "x", |
| 125 | /* 1 */ "y", |
| 126 | /* 2 */ "z", |
| 127 | /* 3 */ "w", |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 128 | }; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 129 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 130 | slangAssert((Index < (sizeof(VectorAccessorMap) / sizeof(const char *))) && |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 131 | "Out-of-bound index to access vector member"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 132 | |
| 133 | return VectorAccessorMap[Index]; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 134 | } |
| 135 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 136 | static const char *GetPackerAPIName(const RSExportPrimitiveType *EPT) { |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 137 | static const char *PrimitiveTypePackerAPINameMap[] = { |
Pirama Arumuga Nainar | e4dd17d | 2015-04-07 11:09:14 -0700 | [diff] [blame] | 138 | "addF16", // DataTypeFloat16 |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 139 | "addF32", // DataTypeFloat32 |
| 140 | "addF64", // DataTypeFloat64 |
| 141 | "addI8", // DataTypeSigned8 |
| 142 | "addI16", // DataTypeSigned16 |
| 143 | "addI32", // DataTypeSigned32 |
| 144 | "addI64", // DataTypeSigned64 |
| 145 | "addU8", // DataTypeUnsigned8 |
| 146 | "addU16", // DataTypeUnsigned16 |
| 147 | "addU32", // DataTypeUnsigned32 |
| 148 | "addU64", // DataTypeUnsigned64 |
| 149 | "addBoolean", // DataTypeBoolean |
| 150 | "addU16", // DataTypeUnsigned565 |
| 151 | "addU16", // DataTypeUnsigned5551 |
| 152 | "addU16", // DataTypeUnsigned4444 |
| 153 | "addMatrix", // DataTypeRSMatrix2x2 |
| 154 | "addMatrix", // DataTypeRSMatrix3x3 |
| 155 | "addMatrix", // DataTypeRSMatrix4x4 |
| 156 | "addObj", // DataTypeRSElement |
| 157 | "addObj", // DataTypeRSType |
| 158 | "addObj", // DataTypeRSAllocation |
| 159 | "addObj", // DataTypeRSSampler |
| 160 | "addObj", // DataTypeRSScript |
| 161 | "addObj", // DataTypeRSMesh |
| 162 | "addObj", // DataTypeRSPath |
| 163 | "addObj", // DataTypeRSProgramFragment |
| 164 | "addObj", // DataTypeRSProgramVertex |
| 165 | "addObj", // DataTypeRSProgramRaster |
| 166 | "addObj", // DataTypeRSProgramStore |
| 167 | "addObj", // DataTypeRSFont |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 168 | }; |
| 169 | unsigned TypeId = EPT->getType(); |
| 170 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 171 | if (TypeId < (sizeof(PrimitiveTypePackerAPINameMap) / sizeof(const char *))) |
| 172 | return PrimitiveTypePackerAPINameMap[EPT->getType()]; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 173 | |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 174 | slangAssert(false && "GetPackerAPIName : Unknown primitive data type"); |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 175 | return nullptr; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 176 | } |
| 177 | |
Stephen Hines | d369cda | 2012-02-13 12:00:03 -0800 | [diff] [blame] | 178 | static std::string GetTypeName(const RSExportType *ET, bool Brackets = true) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 179 | switch (ET->getClass()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 180 | case RSExportType::ExportClassPrimitive: { |
| 181 | return RSExportPrimitiveType::getRSReflectionType( |
| 182 | static_cast<const RSExportPrimitiveType *>(ET))->java_name; |
| 183 | } |
| 184 | case RSExportType::ExportClassPointer: { |
| 185 | const RSExportType *PointeeType = |
| 186 | static_cast<const RSExportPointerType *>(ET)->getPointeeType(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 187 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 188 | if (PointeeType->getClass() != RSExportType::ExportClassRecord) |
| 189 | return "Allocation"; |
| 190 | else |
| 191 | return PointeeType->getElementName(); |
| 192 | } |
| 193 | case RSExportType::ExportClassVector: { |
| 194 | const RSExportVectorType *EVT = static_cast<const RSExportVectorType *>(ET); |
| 195 | std::stringstream VecName; |
| 196 | VecName << EVT->getRSReflectionType(EVT)->rs_java_vector_prefix |
| 197 | << EVT->getNumElement(); |
| 198 | return VecName.str(); |
| 199 | } |
| 200 | case RSExportType::ExportClassMatrix: { |
| 201 | return GetMatrixTypeName(static_cast<const RSExportMatrixType *>(ET)); |
| 202 | } |
| 203 | case RSExportType::ExportClassConstantArray: { |
| 204 | const RSExportConstantArrayType *CAT = |
| 205 | static_cast<const RSExportConstantArrayType *>(ET); |
| 206 | std::string ElementTypeName = GetTypeName(CAT->getElementType()); |
| 207 | if (Brackets) { |
| 208 | ElementTypeName.append("[]"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 209 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 210 | return ElementTypeName; |
| 211 | } |
| 212 | case RSExportType::ExportClassRecord: { |
| 213 | return ET->getElementName() + "." RS_TYPE_ITEM_CLASS_NAME; |
| 214 | } |
| 215 | default: { slangAssert(false && "Unknown class of type"); } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 216 | } |
| 217 | |
| 218 | return ""; |
| 219 | } |
| 220 | |
Shih-wei Liao | cf950c4 | 2010-10-06 03:44:40 -0700 | [diff] [blame] | 221 | static const char *GetTypeNullValue(const RSExportType *ET) { |
| 222 | switch (ET->getClass()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 223 | case RSExportType::ExportClassPrimitive: { |
| 224 | const RSExportPrimitiveType *EPT = |
| 225 | static_cast<const RSExportPrimitiveType *>(ET); |
| 226 | if (EPT->isRSObjectType()) |
Shih-wei Liao | cf950c4 | 2010-10-06 03:44:40 -0700 | [diff] [blame] | 227 | return "null"; |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 228 | else if (EPT->getType() == DataTypeBoolean) |
| 229 | return "false"; |
| 230 | else |
| 231 | return "0"; |
| 232 | break; |
| 233 | } |
| 234 | case RSExportType::ExportClassPointer: |
| 235 | case RSExportType::ExportClassVector: |
| 236 | case RSExportType::ExportClassMatrix: |
| 237 | case RSExportType::ExportClassConstantArray: |
| 238 | case RSExportType::ExportClassRecord: { |
| 239 | return "null"; |
| 240 | break; |
| 241 | } |
| 242 | default: { slangAssert(false && "Unknown class of type"); } |
Shih-wei Liao | cf950c4 | 2010-10-06 03:44:40 -0700 | [diff] [blame] | 243 | } |
| 244 | return ""; |
| 245 | } |
| 246 | |
Stephen Hines | 47aca4e | 2012-03-08 20:07:28 -0800 | [diff] [blame] | 247 | static std::string GetBuiltinElementConstruct(const RSExportType *ET) { |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 248 | if (ET->getClass() == RSExportType::ExportClassPrimitive) { |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 249 | return std::string("Element.") + ET->getElementName(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 250 | } else if (ET->getClass() == RSExportType::ExportClassVector) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 251 | const RSExportVectorType *EVT = static_cast<const RSExportVectorType *>(ET); |
Jean-Luc Brouillet | cec9b65 | 2014-05-14 19:33:57 -0700 | [diff] [blame] | 252 | if (EVT->getType() == DataTypeFloat32) { |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 253 | if (EVT->getNumElement() == 2) { |
Stephen Hines | 2b8fb64 | 2012-03-09 00:12:47 -0800 | [diff] [blame] | 254 | return "Element.F32_2"; |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 255 | } else if (EVT->getNumElement() == 3) { |
Stephen Hines | 2b8fb64 | 2012-03-09 00:12:47 -0800 | [diff] [blame] | 256 | return "Element.F32_3"; |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 257 | } else if (EVT->getNumElement() == 4) { |
Stephen Hines | 2b8fb64 | 2012-03-09 00:12:47 -0800 | [diff] [blame] | 258 | return "Element.F32_4"; |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 259 | } else { |
| 260 | slangAssert(false && "Vectors should be size 2, 3, 4"); |
| 261 | } |
Jean-Luc Brouillet | cec9b65 | 2014-05-14 19:33:57 -0700 | [diff] [blame] | 262 | } else if (EVT->getType() == DataTypeUnsigned8) { |
Stephen Hines | 2b8fb64 | 2012-03-09 00:12:47 -0800 | [diff] [blame] | 263 | if (EVT->getNumElement() == 4) |
| 264 | return "Element.U8_4"; |
Shih-wei Liao | 324c047 | 2010-06-21 13:15:11 -0700 | [diff] [blame] | 265 | } |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 266 | } else if (ET->getClass() == RSExportType::ExportClassMatrix) { |
| 267 | const RSExportMatrixType *EMT = static_cast<const RSExportMatrixType *>(ET); |
| 268 | switch (EMT->getDim()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 269 | case 2: |
| 270 | return "Element.MATRIX_2X2"; |
| 271 | case 3: |
| 272 | return "Element.MATRIX_3X3"; |
| 273 | case 4: |
| 274 | return "Element.MATRIX_4X4"; |
| 275 | default: |
| 276 | slangAssert(false && "Unsupported dimension of matrix"); |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 277 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 278 | } |
Stephen Hines | 47aca4e | 2012-03-08 20:07:28 -0800 | [diff] [blame] | 279 | // RSExportType::ExportClassPointer can't be generated in a struct. |
Shih-wei Liao | 324c047 | 2010-06-21 13:15:11 -0700 | [diff] [blame] | 280 | |
Stephen Hines | 47aca4e | 2012-03-08 20:07:28 -0800 | [diff] [blame] | 281 | return ""; |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 282 | } |
| 283 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 284 | // If FromIntegerType == DestIntegerType, then Value is returned. |
| 285 | // Otherwise, return a Java expression that zero-extends the value |
| 286 | // Value, assumed to be of type FromIntegerType, to the integer type |
| 287 | // DestIntegerType. |
| 288 | // |
| 289 | // Intended operations: |
| 290 | // byte -> {byte,int,short,long} |
| 291 | // short -> {short,int,long} |
| 292 | // int -> {int,long} |
| 293 | // long -> long |
| 294 | static std::string ZeroExtendValue(const std::string &Value, |
| 295 | const std::string &FromIntegerType, |
| 296 | const std::string &DestIntegerType) { |
| 297 | #ifndef __DISABLE_ASSERTS |
| 298 | // Integer types arranged in increasing order by width |
| 299 | const std::vector<std::string> ValidTypes{"byte", "short", "int", "long"}; |
| 300 | auto FromTypeLoc = std::find(ValidTypes.begin(), ValidTypes.end(), FromIntegerType); |
| 301 | auto DestTypeLoc = std::find(ValidTypes.begin(), ValidTypes.end(), DestIntegerType); |
| 302 | // Check that both types are valid. |
| 303 | slangAssert(FromTypeLoc != ValidTypes.end()); |
| 304 | slangAssert(DestTypeLoc != ValidTypes.end()); |
| 305 | // Check that DestIntegerType is at least as wide as FromIntegerType. |
| 306 | slangAssert(FromTypeLoc - ValidTypes.begin() <= DestTypeLoc - ValidTypes.begin()); |
| 307 | #endif |
| 308 | |
| 309 | if (FromIntegerType == DestIntegerType) { |
| 310 | return Value; |
| 311 | } |
| 312 | |
| 313 | std::string Mask, MaskLiteralType; |
| 314 | if (FromIntegerType == "byte") { |
| 315 | Mask = "0xff"; |
| 316 | MaskLiteralType = "int"; |
| 317 | } else if (FromIntegerType == "short") { |
| 318 | Mask = "0xffff"; |
| 319 | MaskLiteralType = "int"; |
| 320 | } else if (FromIntegerType == "int") { |
| 321 | Mask = "0xffffffffL"; |
| 322 | MaskLiteralType = "long"; |
| 323 | } else { |
| 324 | // long -> long casts should have already been handled. |
| 325 | slangAssert(false && "Unknown integer type"); |
| 326 | } |
| 327 | |
| 328 | // Cast the mask to the appropriate type. |
| 329 | if (MaskLiteralType != DestIntegerType) { |
| 330 | Mask = "(" + DestIntegerType + ") " + Mask; |
| 331 | } |
| 332 | return "((" + DestIntegerType + ") ((" + Value + ") & " + Mask + "))"; |
| 333 | } |
| 334 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 335 | /********************** Methods to generate script class **********************/ |
Jean-Luc Brouillet | 59f22c3 | 2014-06-04 14:53:48 -0700 | [diff] [blame] | 336 | RSReflectionJava::RSReflectionJava(const RSContext *Context, |
| 337 | std::vector<std::string> *GeneratedFileNames, |
| 338 | const std::string &OutputBaseDirectory, |
| 339 | const std::string &RSSourceFileName, |
| 340 | const std::string &BitCodeFileName, |
| 341 | bool EmbedBitcodeInJava) |
| 342 | : mRSContext(Context), mPackageName(Context->getReflectJavaPackageName()), |
| 343 | mRSPackageName(Context->getRSPackageName()), |
| 344 | mOutputBaseDirectory(OutputBaseDirectory), |
| 345 | mRSSourceFileName(RSSourceFileName), mBitCodeFileName(BitCodeFileName), |
| 346 | mResourceId(RSSlangReflectUtils::JavaClassNameFromRSFileName( |
| 347 | mBitCodeFileName.c_str())), |
| 348 | mScriptClassName(RS_SCRIPT_CLASS_NAME_PREFIX + |
| 349 | RSSlangReflectUtils::JavaClassNameFromRSFileName( |
| 350 | mRSSourceFileName.c_str())), |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 351 | mEmbedBitcodeInJava(EmbedBitcodeInJava), mNextExportVarSlot(0), |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 352 | mNextExportFuncSlot(0), mNextExportForEachSlot(0), |
| 353 | mNextExportReduceSlot(0), mLastError(""), |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 354 | mGeneratedFileNames(GeneratedFileNames), mFieldIndex(0) { |
Jean-Luc Brouillet | 59f22c3 | 2014-06-04 14:53:48 -0700 | [diff] [blame] | 355 | slangAssert(mGeneratedFileNames && "Must supply GeneratedFileNames"); |
| 356 | slangAssert(!mPackageName.empty() && mPackageName != "-"); |
| 357 | |
| 358 | mOutputDirectory = RSSlangReflectUtils::ComputePackagedPath( |
| 359 | OutputBaseDirectory.c_str(), mPackageName.c_str()) + |
| 360 | OS_PATH_SEPARATOR_STR; |
Tim Murray | 3a38b74 | 2014-07-02 10:41:08 -0700 | [diff] [blame] | 361 | |
| 362 | // mElement.getBytesSize only exists on JB+ |
| 363 | if (mRSContext->getTargetAPI() >= SLANG_JB_TARGET_API) { |
| 364 | mItemSizeof = RS_TYPE_ITEM_SIZEOF_CURRENT; |
| 365 | } else { |
| 366 | mItemSizeof = RS_TYPE_ITEM_SIZEOF_LEGACY; |
| 367 | } |
Jean-Luc Brouillet | 59f22c3 | 2014-06-04 14:53:48 -0700 | [diff] [blame] | 368 | } |
| 369 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 370 | bool RSReflectionJava::genScriptClass(const std::string &ClassName, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 371 | std::string &ErrorMsg) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 372 | if (!startClass(AM_Public, false, ClassName, RS_SCRIPT_CLASS_SUPER_CLASS_NAME, |
| 373 | ErrorMsg)) |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 374 | return false; |
| 375 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 376 | genScriptClassConstructor(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 377 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 378 | // Reflect exported variables |
| 379 | for (auto I = mRSContext->export_vars_begin(), |
| 380 | E = mRSContext->export_vars_end(); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 381 | I != E; I++) |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 382 | genExportVariable(*I); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 383 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 384 | // Reflect exported forEach functions (only available on ICS+) |
Stephen Hines | 4cc499d | 2011-08-24 19:06:17 -0700 | [diff] [blame] | 385 | if (mRSContext->getTargetAPI() >= SLANG_ICS_TARGET_API) { |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 386 | for (auto I = mRSContext->export_foreach_begin(), |
| 387 | E = mRSContext->export_foreach_end(); |
| 388 | I != E; I++) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 389 | genExportForEach(*I); |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 390 | } |
Stephen Hines | 4a4bf92 | 2011-08-18 17:20:33 -0700 | [diff] [blame] | 391 | } |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 392 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 393 | // Reflect exported reduce functions |
| 394 | for (auto I = mRSContext->export_reduce_begin(), |
| 395 | E = mRSContext->export_reduce_end(); |
| 396 | I != E; ++I) |
| 397 | genExportReduce(*I); |
| 398 | |
| 399 | // Reflect exported functions (invokable) |
| 400 | for (auto I = mRSContext->export_funcs_begin(), |
| 401 | E = mRSContext->export_funcs_end(); |
| 402 | I != E; ++I) |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 403 | genExportFunction(*I); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 404 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 405 | endClass(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 406 | |
| 407 | return true; |
| 408 | } |
| 409 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 410 | void RSReflectionJava::genScriptClassConstructor() { |
Stephen Hines | 4c8b659 | 2014-05-22 20:18:45 -0700 | [diff] [blame] | 411 | std::string className(RSSlangReflectUtils::JavaBitcodeClassNameFromRSFileName( |
Jean-Luc Brouillet | 59f22c3 | 2014-06-04 14:53:48 -0700 | [diff] [blame] | 412 | mRSSourceFileName.c_str())); |
Stephen Hines | d293693 | 2012-09-12 20:44:32 -0700 | [diff] [blame] | 413 | // Provide a simple way to reference this object. |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 414 | mOut.indent() << "private static final String " RS_RESOURCE_NAME " = \"" |
| 415 | << getResourceId() << "\";\n"; |
Stephen Hines | d293693 | 2012-09-12 20:44:32 -0700 | [diff] [blame] | 416 | |
| 417 | // Generate a simple constructor with only a single parameter (the rest |
| 418 | // can be inferred from information we already have). |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 419 | mOut.indent() << "// Constructor\n"; |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 420 | startFunction(AM_Public, false, nullptr, getClassName(), 1, "RenderScript", |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 421 | "rs"); |
Stephen Hines | 44d495d | 2014-05-22 19:42:55 -0700 | [diff] [blame] | 422 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 423 | const bool haveReduceExportables = |
| 424 | mRSContext->export_reduce_begin() != mRSContext->export_reduce_end(); |
| 425 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 426 | if (getEmbedBitcodeInJava()) { |
Stephen Hines | 4c8b659 | 2014-05-22 20:18:45 -0700 | [diff] [blame] | 427 | // Call new single argument Java-only constructor |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 428 | mOut.indent() << "super(rs,\n"; |
| 429 | mOut.indent() << " " << RS_RESOURCE_NAME ",\n"; |
| 430 | mOut.indent() << " " << className << ".getBitCode32(),\n"; |
Stephen Hines | 9ae18b2 | 2014-06-10 23:53:00 -0700 | [diff] [blame] | 431 | mOut.indent() << " " << className << ".getBitCode64());\n"; |
Stephen Hines | 4c8b659 | 2014-05-22 20:18:45 -0700 | [diff] [blame] | 432 | } else { |
| 433 | // Call alternate constructor with required parameters. |
| 434 | // Look up the proper raw bitcode resource id via the context. |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 435 | mOut.indent() << "this(rs,\n"; |
| 436 | mOut.indent() << " rs.getApplicationContext().getResources(),\n"; |
| 437 | mOut.indent() << " rs.getApplicationContext().getResources()." |
| 438 | "getIdentifier(\n"; |
| 439 | mOut.indent() << " " RS_RESOURCE_NAME ", \"raw\",\n"; |
| 440 | mOut.indent() |
| 441 | << " rs.getApplicationContext().getPackageName()));\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 442 | endFunction(); |
Stephen Hines | d293693 | 2012-09-12 20:44:32 -0700 | [diff] [blame] | 443 | |
Stephen Hines | 4c8b659 | 2014-05-22 20:18:45 -0700 | [diff] [blame] | 444 | // Alternate constructor (legacy) with 3 original parameters. |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 445 | startFunction(AM_Public, false, nullptr, getClassName(), 3, "RenderScript", |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 446 | "rs", "Resources", "resources", "int", "id"); |
Stephen Hines | 4c8b659 | 2014-05-22 20:18:45 -0700 | [diff] [blame] | 447 | // Call constructor of super class |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 448 | mOut.indent() << "super(rs, resources, id);\n"; |
Stephen Hines | 4c8b659 | 2014-05-22 20:18:45 -0700 | [diff] [blame] | 449 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 450 | |
| 451 | // If an exported variable has initial value, reflect it |
| 452 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 453 | for (auto I = mRSContext->export_vars_begin(), |
| 454 | E = mRSContext->export_vars_end(); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 455 | I != E; I++) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 456 | const RSExportVar *EV = *I; |
Stephen Hines | d369cda | 2012-02-13 12:00:03 -0800 | [diff] [blame] | 457 | if (!EV->getInit().isUninit()) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 458 | genInitExportVariable(EV->getType(), EV->getName(), EV->getInit()); |
Stephen Hines | d369cda | 2012-02-13 12:00:03 -0800 | [diff] [blame] | 459 | } else if (EV->getArraySize()) { |
| 460 | // Always create an initial zero-init array object. |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 461 | mOut.indent() << RS_EXPORT_VAR_PREFIX << EV->getName() << " = new " |
| 462 | << GetTypeName(EV->getType(), false) << "[" |
| 463 | << EV->getArraySize() << "];\n"; |
Stephen Hines | d369cda | 2012-02-13 12:00:03 -0800 | [diff] [blame] | 464 | size_t NumInits = EV->getNumInits(); |
| 465 | const RSExportConstantArrayType *ECAT = |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 466 | static_cast<const RSExportConstantArrayType *>(EV->getType()); |
Stephen Hines | d369cda | 2012-02-13 12:00:03 -0800 | [diff] [blame] | 467 | const RSExportType *ET = ECAT->getElementType(); |
| 468 | for (size_t i = 0; i < NumInits; i++) { |
| 469 | std::stringstream Name; |
| 470 | Name << EV->getName() << "[" << i << "]"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 471 | genInitExportVariable(ET, Name.str(), EV->getInitArray(i)); |
Stephen Hines | d369cda | 2012-02-13 12:00:03 -0800 | [diff] [blame] | 472 | } |
| 473 | } |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 474 | if (mRSContext->getTargetAPI() >= SLANG_JB_TARGET_API) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 475 | genTypeInstance(EV->getType()); |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 476 | } |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 477 | genFieldPackerInstance(EV->getType()); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 478 | } |
| 479 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 480 | if (haveReduceExportables) { |
| 481 | mOut.indent() << SAVED_RS_REFERENCE << " = rs;\n"; |
| 482 | } |
| 483 | |
| 484 | // Reflect argument / return types in kernels |
| 485 | |
| 486 | for (auto I = mRSContext->export_foreach_begin(), |
| 487 | E = mRSContext->export_foreach_end(); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 488 | I != E; I++) { |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 489 | const RSExportForEach *EF = *I; |
| 490 | |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 491 | const RSExportForEach::InTypeVec &InTypes = EF->getInTypes(); |
| 492 | for (RSExportForEach::InTypeIter BI = InTypes.begin(), EI = InTypes.end(); |
| 493 | BI != EI; BI++) { |
| 494 | |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 495 | if (*BI != nullptr) { |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 496 | genTypeInstanceFromPointer(*BI); |
| 497 | } |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 498 | } |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 499 | |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 500 | const RSExportType *OET = EF->getOutType(); |
| 501 | if (OET) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 502 | genTypeInstanceFromPointer(OET); |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 503 | } |
| 504 | } |
| 505 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 506 | for (auto I = mRSContext->export_reduce_begin(), |
| 507 | E = mRSContext->export_reduce_end(); |
| 508 | I != E; I++) { |
| 509 | const RSExportReduce *ER = *I; |
| 510 | genTypeInstance(ER->getType()); |
| 511 | } |
| 512 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 513 | endFunction(); |
Jason Sams | b6902e2 | 2010-11-03 17:04:06 -0700 | [diff] [blame] | 514 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 515 | for (std::set<std::string>::iterator I = mTypesToCheck.begin(), |
| 516 | E = mTypesToCheck.end(); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 517 | I != E; I++) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 518 | mOut.indent() << "private Element " RS_ELEM_PREFIX << *I << ";\n"; |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 519 | } |
| 520 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 521 | for (std::set<std::string>::iterator I = mFieldPackerTypes.begin(), |
| 522 | E = mFieldPackerTypes.end(); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 523 | I != E; I++) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 524 | mOut.indent() << "private FieldPacker " RS_FP_PREFIX << *I << ";\n"; |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 525 | } |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 526 | |
| 527 | if (haveReduceExportables) { |
| 528 | // We save a private copy of rs in order to create temporary |
| 529 | // allocations in the reduce_* entry points. |
| 530 | mOut.indent() << "private RenderScript " << SAVED_RS_REFERENCE << ";\n"; |
| 531 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 532 | } |
| 533 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 534 | void RSReflectionJava::genInitBoolExportVariable(const std::string &VarName, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 535 | const clang::APValue &Val) { |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 536 | slangAssert(!Val.isUninit() && "Not a valid initializer"); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 537 | slangAssert((Val.getKind() == clang::APValue::Int) && |
| 538 | "Bool type has wrong initial APValue"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 539 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 540 | mOut.indent() << RS_EXPORT_VAR_PREFIX << VarName << " = "; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 541 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 542 | mOut << ((Val.getInt().getSExtValue() == 0) ? "false" : "true") << ";\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 543 | } |
| 544 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 545 | void |
| 546 | RSReflectionJava::genInitPrimitiveExportVariable(const std::string &VarName, |
| 547 | const clang::APValue &Val) { |
Stephen Hines | 5d67178 | 2012-01-31 19:32:04 -0800 | [diff] [blame] | 548 | slangAssert(!Val.isUninit() && "Not a valid initializer"); |
| 549 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 550 | mOut.indent() << RS_EXPORT_VAR_PREFIX << VarName << " = "; |
Jean-Luc Brouillet | efcff10 | 2014-06-03 16:13:51 -0700 | [diff] [blame] | 551 | genInitValue(Val, false); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 552 | mOut << ";\n"; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 553 | } |
| 554 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 555 | void RSReflectionJava::genInitExportVariable(const RSExportType *ET, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 556 | const std::string &VarName, |
| 557 | const clang::APValue &Val) { |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 558 | slangAssert(!Val.isUninit() && "Not a valid initializer"); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 559 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 560 | switch (ET->getClass()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 561 | case RSExportType::ExportClassPrimitive: { |
| 562 | const RSExportPrimitiveType *EPT = |
| 563 | static_cast<const RSExportPrimitiveType *>(ET); |
| 564 | if (EPT->getType() == DataTypeBoolean) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 565 | genInitBoolExportVariable(VarName, Val); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 566 | } else { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 567 | genInitPrimitiveExportVariable(VarName, Val); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 568 | } |
| 569 | break; |
| 570 | } |
| 571 | case RSExportType::ExportClassPointer: { |
| 572 | if (!Val.isInt() || Val.getInt().getSExtValue() != 0) |
| 573 | std::cout << "Initializer which is non-NULL to pointer type variable " |
| 574 | "will be ignored\n"; |
| 575 | break; |
| 576 | } |
| 577 | case RSExportType::ExportClassVector: { |
| 578 | const RSExportVectorType *EVT = static_cast<const RSExportVectorType *>(ET); |
| 579 | switch (Val.getKind()) { |
| 580 | case clang::APValue::Int: |
| 581 | case clang::APValue::Float: { |
| 582 | for (unsigned i = 0; i < EVT->getNumElement(); i++) { |
| 583 | std::string Name = VarName + "." + GetVectorAccessor(i); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 584 | genInitPrimitiveExportVariable(Name, Val); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 585 | } |
| 586 | break; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 587 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 588 | case clang::APValue::Vector: { |
| 589 | std::stringstream VecName; |
| 590 | VecName << EVT->getRSReflectionType(EVT)->rs_java_vector_prefix |
| 591 | << EVT->getNumElement(); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 592 | mOut.indent() << RS_EXPORT_VAR_PREFIX << VarName << " = new " |
| 593 | << VecName.str() << "();\n"; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 594 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 595 | unsigned NumElements = std::min( |
| 596 | static_cast<unsigned>(EVT->getNumElement()), Val.getVectorLength()); |
| 597 | for (unsigned i = 0; i < NumElements; i++) { |
| 598 | const clang::APValue &ElementVal = Val.getVectorElt(i); |
| 599 | std::string Name = VarName + "." + GetVectorAccessor(i); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 600 | genInitPrimitiveExportVariable(Name, ElementVal); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 601 | } |
| 602 | break; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 603 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 604 | case clang::APValue::MemberPointer: |
| 605 | case clang::APValue::Uninitialized: |
| 606 | case clang::APValue::ComplexInt: |
| 607 | case clang::APValue::ComplexFloat: |
| 608 | case clang::APValue::LValue: |
| 609 | case clang::APValue::Array: |
| 610 | case clang::APValue::Struct: |
| 611 | case clang::APValue::Union: |
| 612 | case clang::APValue::AddrLabelDiff: { |
| 613 | slangAssert(false && "Unexpected type of value of initializer."); |
| 614 | } |
| 615 | } |
| 616 | break; |
| 617 | } |
| 618 | // TODO(zonr): Resolving initializer of a record (and matrix) type variable |
| 619 | // is complex. It cannot obtain by just simply evaluating the initializer |
| 620 | // expression. |
| 621 | case RSExportType::ExportClassMatrix: |
| 622 | case RSExportType::ExportClassConstantArray: |
| 623 | case RSExportType::ExportClassRecord: { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 624 | #if 0 |
| 625 | unsigned InitIndex = 0; |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 626 | const RSExportRecordType *ERT = |
| 627 | static_cast<const RSExportRecordType*>(ET); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 628 | |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 629 | slangAssert((Val.getKind() == clang::APValue::Vector) && |
| 630 | "Unexpected type of initializer for record type variable"); |
Shih-wei Liao | 9c631ff | 2010-09-17 11:57:29 -0700 | [diff] [blame] | 631 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 632 | mOut.indent() << RS_EXPORT_VAR_PREFIX << VarName |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 633 | << " = new " << ERT->getElementName() |
Jean-Luc Brouillet | 2968921 | 2014-05-27 13:25:31 -0700 | [diff] [blame] | 634 | << "." RS_TYPE_ITEM_CLASS_NAME"();\n"; |
Shih-wei Liao | 9c631ff | 2010-09-17 11:57:29 -0700 | [diff] [blame] | 635 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 636 | for (RSExportRecordType::const_field_iterator I = ERT->fields_begin(), |
| 637 | E = ERT->fields_end(); |
| 638 | I != E; |
| 639 | I++) { |
| 640 | const RSExportRecordType::Field *F = *I; |
| 641 | std::string FieldName = VarName + "." + F->getName(); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 642 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 643 | if (InitIndex > Val.getVectorLength()) |
| 644 | break; |
| 645 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 646 | genInitPrimitiveExportVariable(FieldName, |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 647 | Val.getVectorElt(InitIndex++)); |
| 648 | } |
| 649 | #endif |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 650 | slangAssert(false && "Unsupported initializer for record/matrix/constant " |
| 651 | "array type variable currently"); |
| 652 | break; |
| 653 | } |
| 654 | default: { slangAssert(false && "Unknown class of type"); } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 655 | } |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 656 | } |
| 657 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 658 | void RSReflectionJava::genExportVariable(const RSExportVar *EV) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 659 | const RSExportType *ET = EV->getType(); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 660 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 661 | mOut.indent() << "private final static int " << RS_EXPORT_VAR_INDEX_PREFIX |
| 662 | << EV->getName() << " = " << getNextExportVarSlot() << ";\n"; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 663 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 664 | switch (ET->getClass()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 665 | case RSExportType::ExportClassPrimitive: { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 666 | genPrimitiveTypeExportVariable(EV); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 667 | break; |
| 668 | } |
| 669 | case RSExportType::ExportClassPointer: { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 670 | genPointerTypeExportVariable(EV); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 671 | break; |
| 672 | } |
| 673 | case RSExportType::ExportClassVector: { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 674 | genVectorTypeExportVariable(EV); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 675 | break; |
| 676 | } |
| 677 | case RSExportType::ExportClassMatrix: { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 678 | genMatrixTypeExportVariable(EV); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 679 | break; |
| 680 | } |
| 681 | case RSExportType::ExportClassConstantArray: { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 682 | genConstantArrayTypeExportVariable(EV); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 683 | break; |
| 684 | } |
| 685 | case RSExportType::ExportClassRecord: { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 686 | genRecordTypeExportVariable(EV); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 687 | break; |
| 688 | } |
| 689 | default: { slangAssert(false && "Unknown class of type"); } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 690 | } |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 691 | } |
| 692 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 693 | void RSReflectionJava::genExportFunction(const RSExportFunc *EF) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 694 | mOut.indent() << "private final static int " << RS_EXPORT_FUNC_INDEX_PREFIX |
| 695 | << EF->getName() << " = " << getNextExportFuncSlot() << ";\n"; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 696 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 697 | // invoke_*() |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 698 | ArgTy Args; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 699 | |
Zonr Chang | 0da0a7d | 2010-10-05 21:26:37 +0800 | [diff] [blame] | 700 | if (EF->hasParam()) { |
| 701 | for (RSExportFunc::const_param_iterator I = EF->params_begin(), |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 702 | E = EF->params_end(); |
| 703 | I != E; I++) { |
| 704 | Args.push_back( |
| 705 | std::make_pair(GetTypeName((*I)->getType()), (*I)->getName())); |
Zonr Chang | 0da0a7d | 2010-10-05 21:26:37 +0800 | [diff] [blame] | 706 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 707 | } |
| 708 | |
Stephen Hines | bd0a7dd | 2015-07-01 19:24:00 -0700 | [diff] [blame] | 709 | if (mRSContext->getTargetAPI() >= SLANG_M_TARGET_API) { |
Yang Ni | cec2a1d | 2014-12-02 10:23:47 -0800 | [diff] [blame] | 710 | startFunction(AM_Public, false, "Script.InvokeID", |
| 711 | "getInvokeID_" + EF->getName(), 0); |
| 712 | |
| 713 | mOut.indent() << "return createInvokeID(" << RS_EXPORT_FUNC_INDEX_PREFIX |
| 714 | << EF->getName() << ");\n"; |
| 715 | |
| 716 | endFunction(); |
| 717 | } |
| 718 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 719 | startFunction(AM_Public, false, "void", |
| 720 | "invoke_" + EF->getName(/*Mangle=*/false), |
| 721 | // We are using un-mangled name since Java |
| 722 | // supports method overloading. |
| 723 | Args); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 724 | |
| 725 | if (!EF->hasParam()) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 726 | mOut.indent() << "invoke(" << RS_EXPORT_FUNC_INDEX_PREFIX << EF->getName() |
| 727 | << ");\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 728 | } else { |
| 729 | const RSExportRecordType *ERT = EF->getParamPacketType(); |
| 730 | std::string FieldPackerName = EF->getName() + "_fp"; |
| 731 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 732 | if (genCreateFieldPacker(ERT, FieldPackerName.c_str())) |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 733 | genPackVarOfType(ERT, nullptr, FieldPackerName.c_str()); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 734 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 735 | mOut.indent() << "invoke(" << RS_EXPORT_FUNC_INDEX_PREFIX << EF->getName() |
| 736 | << ", " << FieldPackerName << ");\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 737 | } |
| 738 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 739 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 740 | } |
| 741 | |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 742 | void RSReflectionJava::genPairwiseDimCheck(std::string name0, |
| 743 | std::string name1) { |
| 744 | |
| 745 | mOut.indent() << "// Verify dimensions\n"; |
| 746 | mOut.indent() << "t0 = " << name0 << ".getType();\n"; |
| 747 | mOut.indent() << "t1 = " << name1 << ".getType();\n"; |
| 748 | mOut.indent() << "if ((t0.getCount() != t1.getCount()) ||\n"; |
| 749 | mOut.indent() << " (t0.getX() != t1.getX()) ||\n"; |
| 750 | mOut.indent() << " (t0.getY() != t1.getY()) ||\n"; |
| 751 | mOut.indent() << " (t0.getZ() != t1.getZ()) ||\n"; |
| 752 | mOut.indent() << " (t0.hasFaces() != t1.hasFaces()) ||\n"; |
| 753 | mOut.indent() << " (t0.hasMipmaps() != t1.hasMipmaps())) {\n"; |
| 754 | mOut.indent() << " throw new RSRuntimeException(\"Dimension mismatch " |
| 755 | << "between parameters " << name0 << " and " << name1 |
| 756 | << "!\");\n"; |
| 757 | mOut.indent() << "}\n\n"; |
| 758 | } |
| 759 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 760 | void RSReflectionJava::genNullOrEmptyArrayCheck(const std::string &ArrayName) { |
| 761 | mOut.indent() << "// Verify that \"" << ArrayName << "\" is non-null.\n"; |
| 762 | mOut.indent() << "if (" << ArrayName << " == null) {\n"; |
| 763 | mOut.indent() << " throw new RSIllegalArgumentException(\"Array \\\"" |
| 764 | << ArrayName << "\\\" is null!\");\n"; |
| 765 | mOut.indent() << "}\n"; |
| 766 | mOut.indent() << "// Verify that \"" << ArrayName << "\" is non-empty.\n"; |
| 767 | mOut.indent() << "if (" << ArrayName << ".length == 0) {\n"; |
| 768 | mOut.indent() << " throw new RSIllegalArgumentException(\"Array \\\"" |
| 769 | << ArrayName << "\\\" is zero-length!\");\n"; |
| 770 | mOut.indent() << "}\n"; |
| 771 | } |
| 772 | |
| 773 | void RSReflectionJava::genVectorLengthCompatibilityCheck(const std::string &ArrayName, |
| 774 | unsigned VecSize) { |
| 775 | mOut.indent() << "// Verify that the array length is a multiple of the vector size.\n"; |
| 776 | mOut.indent() << "if (" << ArrayName << ".length % " << std::to_string(VecSize) |
| 777 | << " != 0) {\n"; |
| 778 | mOut.indent() << " throw new RSIllegalArgumentException(\"Array \\\"" << ArrayName |
| 779 | << "\\\" is not a multiple of " << std::to_string(VecSize) |
| 780 | << " in length!\");\n"; |
| 781 | mOut.indent() << "}\n"; |
| 782 | } |
| 783 | |
| 784 | void RSReflectionJava::gen1DCheck(const std::string &Name) { |
| 785 | // TODO: Check that t0.getArrayCount() == 0, when / if this API is |
| 786 | // un-hidden. |
| 787 | mOut.indent() << "Type t0 = " << Name << ".getType();\n"; |
| 788 | mOut.indent() << "// Verify " << Name << " is 1D\n"; |
| 789 | mOut.indent() << "if (t0.getY() != 0 ||\n"; |
| 790 | mOut.indent() << " t0.hasFaces() ||\n"; |
| 791 | mOut.indent() << " t0.hasMipmaps()) {\n"; |
| 792 | mOut.indent() << " throw new RSIllegalArgumentException(\"Parameter " |
| 793 | << Name << " is not 1D!\");\n"; |
| 794 | mOut.indent() << "}\n\n"; |
| 795 | } |
| 796 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 797 | void RSReflectionJava::genExportForEach(const RSExportForEach *EF) { |
Stephen Hines | c17e198 | 2012-02-22 12:30:45 -0800 | [diff] [blame] | 798 | if (EF->isDummyRoot()) { |
| 799 | // Skip reflection for dummy root() kernels. Note that we have to |
| 800 | // advance the next slot number for ForEach, however. |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 801 | mOut.indent() << "//private final static int " |
| 802 | << RS_EXPORT_FOREACH_INDEX_PREFIX << EF->getName() << " = " |
| 803 | << getNextExportForEachSlot() << ";\n"; |
Stephen Hines | c17e198 | 2012-02-22 12:30:45 -0800 | [diff] [blame] | 804 | return; |
| 805 | } |
| 806 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 807 | mOut.indent() << "private final static int " << RS_EXPORT_FOREACH_INDEX_PREFIX |
| 808 | << EF->getName() << " = " << getNextExportForEachSlot() |
| 809 | << ";\n"; |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 810 | |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 811 | // forEach_*() |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 812 | ArgTy Args; |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 813 | bool HasAllocation = false; // at least one in/out allocation? |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 814 | |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 815 | const RSExportForEach::InVec &Ins = EF->getIns(); |
| 816 | const RSExportForEach::InTypeVec &InTypes = EF->getInTypes(); |
| 817 | const RSExportType *OET = EF->getOutType(); |
| 818 | |
| 819 | if (Ins.size() == 1) { |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 820 | HasAllocation = true; |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 821 | Args.push_back(std::make_pair("Allocation", "ain")); |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 822 | |
| 823 | } else if (Ins.size() > 1) { |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 824 | HasAllocation = true; |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 825 | for (RSExportForEach::InIter BI = Ins.begin(), EI = Ins.end(); BI != EI; |
| 826 | BI++) { |
| 827 | |
| 828 | Args.push_back(std::make_pair("Allocation", |
| 829 | "ain_" + (*BI)->getName().str())); |
| 830 | } |
| 831 | } |
| 832 | |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 833 | if (EF->hasOut() || EF->hasReturn()) { |
| 834 | HasAllocation = true; |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 835 | Args.push_back(std::make_pair("Allocation", "aout")); |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 836 | } |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 837 | |
| 838 | const RSExportRecordType *ERT = EF->getParamPacketType(); |
| 839 | if (ERT) { |
| 840 | for (RSExportForEach::const_param_iterator I = EF->params_begin(), |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 841 | E = EF->params_end(); |
| 842 | I != E; I++) { |
| 843 | Args.push_back( |
| 844 | std::make_pair(GetTypeName((*I)->getType()), (*I)->getName())); |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 845 | } |
| 846 | } |
| 847 | |
Tim Murray | b81a993 | 2012-10-10 15:56:02 -0700 | [diff] [blame] | 848 | if (mRSContext->getTargetAPI() >= SLANG_JB_MR1_TARGET_API) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 849 | startFunction(AM_Public, false, "Script.KernelID", |
| 850 | "getKernelID_" + EF->getName(), 0); |
Tim Murray | b81a993 | 2012-10-10 15:56:02 -0700 | [diff] [blame] | 851 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 852 | // TODO: add element checking |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 853 | mOut.indent() << "return createKernelID(" << RS_EXPORT_FOREACH_INDEX_PREFIX |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 854 | << EF->getName() << ", " << EF->getSignatureMetadata() |
| 855 | << ", null, null);\n"; |
Tim Murray | b81a993 | 2012-10-10 15:56:02 -0700 | [diff] [blame] | 856 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 857 | endFunction(); |
Tim Murray | b81a993 | 2012-10-10 15:56:02 -0700 | [diff] [blame] | 858 | } |
| 859 | |
Stephen Hines | 5097474 | 2013-02-12 22:11:08 -0800 | [diff] [blame] | 860 | if (mRSContext->getTargetAPI() >= SLANG_JB_MR2_TARGET_API) { |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 861 | if (HasAllocation) { |
| 862 | startFunction(AM_Public, false, "void", "forEach_" + EF->getName(), Args); |
Stephen Hines | 5097474 | 2013-02-12 22:11:08 -0800 | [diff] [blame] | 863 | |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 864 | mOut.indent() << "forEach_" << EF->getName(); |
| 865 | mOut << "("; |
Stephen Hines | 5097474 | 2013-02-12 22:11:08 -0800 | [diff] [blame] | 866 | |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 867 | if (Ins.size() == 1) { |
| 868 | mOut << "ain, "; |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 869 | |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 870 | } else if (Ins.size() > 1) { |
| 871 | for (RSExportForEach::InIter BI = Ins.begin(), EI = Ins.end(); BI != EI; |
| 872 | BI++) { |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 873 | |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 874 | mOut << "ain_" << (*BI)->getName().str() << ", "; |
| 875 | } |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 876 | } |
David Gross | fb78d4c | 2015-03-31 13:56:05 -0700 | [diff] [blame] | 877 | |
| 878 | if (EF->hasOut() || EF->hasReturn()) { |
| 879 | mOut << "aout, "; |
| 880 | } |
| 881 | |
| 882 | if (EF->hasUsrData()) { |
| 883 | mOut << Args.back().second << ", "; |
| 884 | } |
| 885 | |
| 886 | // No clipped bounds to pass in. |
| 887 | mOut << "null);\n"; |
| 888 | |
| 889 | endFunction(); |
Stephen Hines | 5097474 | 2013-02-12 22:11:08 -0800 | [diff] [blame] | 890 | } |
| 891 | |
Stephen Hines | 5097474 | 2013-02-12 22:11:08 -0800 | [diff] [blame] | 892 | // Add the clipped kernel parameters to the Args list. |
| 893 | Args.push_back(std::make_pair("Script.LaunchOptions", "sc")); |
| 894 | } |
| 895 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 896 | startFunction(AM_Public, false, "void", "forEach_" + EF->getName(), Args); |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 897 | |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 898 | if (InTypes.size() == 1) { |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 899 | if (InTypes.front() != nullptr) { |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 900 | genTypeCheck(InTypes.front(), "ain"); |
| 901 | } |
| 902 | |
| 903 | } else if (InTypes.size() > 1) { |
| 904 | size_t Index = 0; |
| 905 | for (RSExportForEach::InTypeIter BI = InTypes.begin(), EI = InTypes.end(); |
| 906 | BI != EI; BI++, ++Index) { |
| 907 | |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 908 | if (*BI != nullptr) { |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 909 | genTypeCheck(*BI, ("ain_" + Ins[Index]->getName()).str().c_str()); |
| 910 | } |
| 911 | } |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 912 | } |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 913 | |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 914 | if (OET) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 915 | genTypeCheck(OET, "aout"); |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 916 | } |
| 917 | |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 918 | if (Ins.size() == 1 && (EF->hasOut() || EF->hasReturn())) { |
| 919 | mOut.indent() << "Type t0, t1;"; |
| 920 | genPairwiseDimCheck("ain", "aout"); |
| 921 | |
| 922 | } else if (Ins.size() > 1) { |
| 923 | mOut.indent() << "Type t0, t1;"; |
| 924 | |
| 925 | std::string In0Name = "ain_" + Ins[0]->getName().str(); |
| 926 | |
| 927 | for (size_t index = 1; index < Ins.size(); ++index) { |
| 928 | genPairwiseDimCheck(In0Name, "ain_" + Ins[index]->getName().str()); |
| 929 | } |
| 930 | |
| 931 | if (EF->hasOut() || EF->hasReturn()) { |
| 932 | genPairwiseDimCheck(In0Name, "aout"); |
| 933 | } |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 934 | } |
| 935 | |
| 936 | std::string FieldPackerName = EF->getName() + "_fp"; |
| 937 | if (ERT) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 938 | if (genCreateFieldPacker(ERT, FieldPackerName.c_str())) { |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 939 | genPackVarOfType(ERT, nullptr, FieldPackerName.c_str()); |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 940 | } |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 941 | } |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 942 | mOut.indent() << "forEach(" << RS_EXPORT_FOREACH_INDEX_PREFIX |
| 943 | << EF->getName(); |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 944 | |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 945 | if (Ins.size() == 1) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 946 | mOut << ", ain"; |
Chris Wailes | c9454af | 2014-06-13 17:25:40 -0700 | [diff] [blame] | 947 | } else if (Ins.size() > 1) { |
| 948 | mOut << ", new Allocation[]{ain_" << Ins[0]->getName().str(); |
| 949 | |
| 950 | for (size_t index = 1; index < Ins.size(); ++index) { |
| 951 | mOut << ", ain_" << Ins[index]->getName().str(); |
| 952 | } |
| 953 | |
| 954 | mOut << "}"; |
| 955 | |
| 956 | } else { |
| 957 | mOut << ", (Allocation) null"; |
| 958 | } |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 959 | |
Stephen Hines | 9ca96e7 | 2012-09-13 16:57:06 -0700 | [diff] [blame] | 960 | if (EF->hasOut() || EF->hasReturn()) |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 961 | mOut << ", aout"; |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 962 | else |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 963 | mOut << ", null"; |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 964 | |
| 965 | if (EF->hasUsrData()) |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 966 | mOut << ", " << FieldPackerName; |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 967 | else |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 968 | mOut << ", null"; |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 969 | |
Stephen Hines | 5097474 | 2013-02-12 22:11:08 -0800 | [diff] [blame] | 970 | if (mRSContext->getTargetAPI() >= SLANG_JB_MR2_TARGET_API) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 971 | mOut << ", sc);\n"; |
Stephen Hines | 5097474 | 2013-02-12 22:11:08 -0800 | [diff] [blame] | 972 | } else { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 973 | mOut << ");\n"; |
Stephen Hines | 5097474 | 2013-02-12 22:11:08 -0800 | [diff] [blame] | 974 | } |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 975 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 976 | endFunction(); |
Stephen Hines | 593a894 | 2011-05-10 15:29:50 -0700 | [diff] [blame] | 977 | } |
| 978 | |
Matt Wala | 7682b66 | 2015-07-30 15:53:47 -0700 | [diff] [blame] | 979 | void RSReflectionJava::genExportReduce(const RSExportReduce *ER) { |
| 980 | // Generate the reflected function index. |
| 981 | mOut.indent() << "private final static int " << RS_EXPORT_REDUCE_INDEX_PREFIX |
| 982 | << ER->getName() << " = " << getNextExportReduceSlot() |
| 983 | << ";\n"; |
| 984 | |
| 985 | // Two variants of reduce_* entry points get generated: |
| 986 | // Array variant: |
| 987 | // ty' reduce_foo(ty[] input) |
| 988 | // ty' reduce_foo(ty[] input, int x1, int x2) |
| 989 | // Allocation variant: |
| 990 | // void reduce_foo(Allocation ain, Allocation aout) |
| 991 | // void reduce_foo(Allocation ain, Allocation aout, Script.LaunchOptions sc) |
| 992 | |
| 993 | const RSExportType *Type = ER->getType(); |
| 994 | const std::string Name = ER->getName(); |
| 995 | |
| 996 | genExportReduceArrayVariant(Type, Name); |
| 997 | genExportReduceAllocationVariant(Type, Name); |
| 998 | } |
| 999 | |
| 1000 | void RSReflectionJava::genExportReduceAllocationVariant(const RSExportType *Type, |
| 1001 | const std::string &KernelName) { |
| 1002 | const std::string FuncName = "reduce_" + KernelName; |
| 1003 | |
| 1004 | // void reduce_foo(Allocation ain, Allocation aout) |
| 1005 | startFunction(AM_Public, false, "void", FuncName, 2, |
| 1006 | "Allocation", "ain", |
| 1007 | "Allocation", "aout"); |
| 1008 | mOut.indent() << FuncName << "(ain, aout, null);\n"; |
| 1009 | endFunction(); |
| 1010 | |
| 1011 | // void reduce_foo(Allocation ain, Allocation aout, Script.LaunchOptions sc) |
| 1012 | startFunction(AM_Public, false, "void", FuncName, 3, |
| 1013 | "Allocation", "ain", |
| 1014 | "Allocation", "aout", |
| 1015 | "Script.LaunchOptions", "sc"); |
| 1016 | |
| 1017 | // Type checking |
| 1018 | genTypeCheck(Type, "ain"); |
| 1019 | genTypeCheck(Type, "aout"); |
| 1020 | |
| 1021 | // Check that the input is 1D |
| 1022 | gen1DCheck("ain"); |
| 1023 | |
| 1024 | // Call backend |
| 1025 | |
| 1026 | // Script.reduce has the signature |
| 1027 | // |
| 1028 | // protected void |
| 1029 | // reduce(int slot, Allocation ain, Allocation aout, Script.LaunchOptions sc) |
| 1030 | mOut.indent() << "reduce(" |
| 1031 | << RS_EXPORT_REDUCE_INDEX_PREFIX << KernelName |
| 1032 | << ", ain, aout, sc);\n"; |
| 1033 | |
| 1034 | endFunction(); |
| 1035 | } |
| 1036 | |
| 1037 | void RSReflectionJava::genExportReduceArrayVariant(const RSExportType *Type, |
| 1038 | const std::string &KernelName) { |
| 1039 | // Determine if the array variant can be generated. Some type |
| 1040 | // classes cannot be reflected in Java. |
| 1041 | auto Class = Type->getClass(); |
| 1042 | if (Class != RSExportType::ExportClassPrimitive && |
| 1043 | Class != RSExportType::ExportClassVector) { |
| 1044 | return; |
| 1045 | } |
| 1046 | |
| 1047 | RSReflectionTypeData TypeData; |
| 1048 | Type->convertToRTD(&TypeData); |
| 1049 | |
| 1050 | // Check if the type supports reading back from an Allocation and |
| 1051 | // returning as a first class Java type. If not, the helper cannot |
| 1052 | // be generated. |
| 1053 | if (!TypeData.type->java_name || !TypeData.type->java_array_element_name || |
| 1054 | (TypeData.vecSize > 1 && !TypeData.type->rs_java_vector_prefix)) { |
| 1055 | return; |
| 1056 | } |
| 1057 | |
| 1058 | const std::string FuncName = "reduce_" + KernelName; |
| 1059 | const std::string TypeName = GetTypeName(Type); |
| 1060 | const std::string ReflectedScalarType = TypeData.type->java_name; |
| 1061 | const std::string ArrayElementType = TypeData.type->java_array_element_name; |
| 1062 | const std::string ArrayType = ArrayElementType + "[]"; |
| 1063 | const std::string ElementName = Type->getElementName(); |
| 1064 | |
| 1065 | const uint32_t VecSize = TypeData.vecSize; |
| 1066 | |
| 1067 | std::string InLength = "in.length"; |
| 1068 | // Adjust the length so that it corresponds to the number of |
| 1069 | // elements in the allocation. |
| 1070 | if (VecSize > 1) { |
| 1071 | InLength += " / " + std::to_string(VecSize); |
| 1072 | } |
| 1073 | |
| 1074 | // TypeName reduce_foo(ArrayElementType[] in) |
| 1075 | startFunction(AM_Public, false, TypeName.c_str(), FuncName, 1, |
| 1076 | ArrayType.c_str(), "in"); |
| 1077 | genNullOrEmptyArrayCheck("in"); |
| 1078 | if (VecSize > 1) { |
| 1079 | genVectorLengthCompatibilityCheck("in", VecSize); |
| 1080 | } |
| 1081 | mOut.indent() << "return " << FuncName << "(in, 0, " |
| 1082 | << InLength << ");\n"; |
| 1083 | endFunction(); |
| 1084 | |
| 1085 | // TypeName reduce_foo(ArrayElementType[] in, int x1, int x2) |
| 1086 | |
| 1087 | startFunction(AM_Public, false, TypeName.c_str(), FuncName, 3, |
| 1088 | ArrayType.c_str(), "in", |
| 1089 | "int", "x1", |
| 1090 | "int", "x2"); |
| 1091 | |
| 1092 | genNullOrEmptyArrayCheck("in"); |
| 1093 | if (VecSize > 1) { |
| 1094 | genVectorLengthCompatibilityCheck("in", VecSize); |
| 1095 | } |
| 1096 | // Check that 0 <= x1 and x1 < x2 and x2 <= InLength |
| 1097 | mOut.indent() << "// Bounds check passed x1 and x2\n"; |
| 1098 | mOut.indent() << "if (x1 < 0 || x1 >= x2 || x2 > " << InLength << ") {\n"; |
| 1099 | mOut.indent() << " throw new RSRuntimeException(" |
| 1100 | << "\"Input bounds are invalid!\");\n"; |
| 1101 | mOut.indent() << "}\n"; |
| 1102 | |
| 1103 | // Create a temporary input allocation. |
| 1104 | mOut.indent() << "Allocation ain = Allocation.createSized(" |
| 1105 | << SAVED_RS_REFERENCE << ", " |
| 1106 | << RS_ELEM_PREFIX << ElementName << ", " |
| 1107 | << "x2 - x1);\n"; |
| 1108 | mOut.indent() << "ain.setAutoPadding(true);\n"; |
| 1109 | mOut.indent() << "ain.copy1DRangeFrom(x1, x2 - x1, in);\n"; |
| 1110 | |
| 1111 | // Create a temporary output allocation. |
| 1112 | mOut.indent() << "Allocation aout = Allocation.createSized(" |
| 1113 | << SAVED_RS_REFERENCE << ", " |
| 1114 | << RS_ELEM_PREFIX << ElementName << ", " |
| 1115 | << "1);\n"; |
| 1116 | mOut.indent() << "aout.setAutoPadding(true);\n"; |
| 1117 | |
| 1118 | mOut.indent() << FuncName << "(ain, aout, null);\n"; |
| 1119 | |
| 1120 | if (VecSize > 1) { |
| 1121 | // An allocation with vector elements is represented as an array |
| 1122 | // of primitives, so we have to extract the output from the |
| 1123 | // element array and rebuild the vector. |
| 1124 | // |
| 1125 | // E.g. for int2 |
| 1126 | // |
| 1127 | // Allocation outArray = new int[2]; |
| 1128 | // aout.copyTo(outArray); |
| 1129 | // int elem0 = outArray[0]; |
| 1130 | // int elem1 = outArray[1]; |
| 1131 | // return new Int2(elem0, elem1); |
| 1132 | |
| 1133 | mOut.indent() << ArrayType << " outArray = new " |
| 1134 | << ArrayElementType << "[" << VecSize << "];\n"; |
| 1135 | |
| 1136 | mOut.indent() << "aout.copy1DRangeTo(0, 1, outArray);\n"; |
| 1137 | |
| 1138 | for (unsigned Elem = 0; Elem < VecSize; ++Elem) { |
| 1139 | mOut.indent() << ReflectedScalarType << " elem" << Elem << " = "; |
| 1140 | std::string Index = "outArray[" + std::to_string(Elem) + "]"; |
| 1141 | |
| 1142 | if (ReflectedScalarType == ArrayElementType) { |
| 1143 | mOut << Index << ";\n"; |
| 1144 | } else { |
| 1145 | mOut << ZeroExtendValue(Index, ArrayElementType, ReflectedScalarType) << ";\n"; |
| 1146 | } |
| 1147 | } |
| 1148 | |
| 1149 | mOut.indent() << "return new " << TypeName << "("; |
| 1150 | for (unsigned Elem = 0; Elem < VecSize; ++Elem) { |
| 1151 | if (Elem > 0) mOut << ", "; |
| 1152 | mOut << "elem" << Elem; |
| 1153 | } |
| 1154 | mOut << ");\n"; |
| 1155 | } else { |
| 1156 | // Scalar handling. |
| 1157 | // |
| 1158 | // E.g. for int |
| 1159 | // Allocation outArray = new int[1]; |
| 1160 | // aout.copyTo(outArray); |
| 1161 | // return outArray[0]; |
| 1162 | mOut.indent() << ArrayType << " outArray = new " << ArrayElementType |
| 1163 | << "[1];\n"; |
| 1164 | mOut.indent() << "aout.copyTo(outArray);\n"; |
| 1165 | |
| 1166 | if (ReflectedScalarType == "boolean") { |
| 1167 | mOut.indent() << "return outArray[0] != 0;\n"; |
| 1168 | } else if (ReflectedScalarType == ArrayElementType) { |
| 1169 | mOut.indent() << "return outArray[0];\n"; |
| 1170 | } else { |
| 1171 | mOut.indent() << "return " |
| 1172 | << ZeroExtendValue("outArray[0]", |
| 1173 | ArrayElementType, |
| 1174 | ReflectedScalarType) |
| 1175 | << ";\n"; |
| 1176 | } |
| 1177 | } |
| 1178 | |
| 1179 | endFunction(); |
| 1180 | } |
| 1181 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1182 | void RSReflectionJava::genTypeInstanceFromPointer(const RSExportType *ET) { |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1183 | if (ET->getClass() == RSExportType::ExportClassPointer) { |
Stephen Hines | 9ca96e7 | 2012-09-13 16:57:06 -0700 | [diff] [blame] | 1184 | // For pointer parameters to original forEach kernels. |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1185 | const RSExportPointerType *EPT = |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1186 | static_cast<const RSExportPointerType *>(ET); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1187 | genTypeInstance(EPT->getPointeeType()); |
Stephen Hines | 9ca96e7 | 2012-09-13 16:57:06 -0700 | [diff] [blame] | 1188 | } else { |
| 1189 | // For handling pass-by-value kernel parameters. |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1190 | genTypeInstance(ET); |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 1191 | } |
| 1192 | } |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1193 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1194 | void RSReflectionJava::genTypeInstance(const RSExportType *ET) { |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 1195 | switch (ET->getClass()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1196 | case RSExportType::ExportClassPrimitive: |
| 1197 | case RSExportType::ExportClassVector: |
| 1198 | case RSExportType::ExportClassConstantArray: { |
| 1199 | std::string TypeName = ET->getElementName(); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1200 | if (addTypeNameForElement(TypeName)) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1201 | mOut.indent() << RS_ELEM_PREFIX << TypeName << " = Element." << TypeName |
| 1202 | << "(rs);\n"; |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1203 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1204 | break; |
| 1205 | } |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 1206 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1207 | case RSExportType::ExportClassRecord: { |
| 1208 | std::string ClassName = ET->getElementName(); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1209 | if (addTypeNameForElement(ClassName)) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1210 | mOut.indent() << RS_ELEM_PREFIX << ClassName << " = " << ClassName |
| 1211 | << ".createElement(rs);\n"; |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 1212 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1213 | break; |
| 1214 | } |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 1215 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1216 | default: |
| 1217 | break; |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1218 | } |
| 1219 | } |
| 1220 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1221 | void RSReflectionJava::genFieldPackerInstance(const RSExportType *ET) { |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 1222 | switch (ET->getClass()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1223 | case RSExportType::ExportClassPrimitive: |
| 1224 | case RSExportType::ExportClassVector: |
| 1225 | case RSExportType::ExportClassConstantArray: |
| 1226 | case RSExportType::ExportClassRecord: { |
| 1227 | std::string TypeName = ET->getElementName(); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1228 | addTypeNameForFieldPacker(TypeName); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1229 | break; |
| 1230 | } |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 1231 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1232 | default: |
| 1233 | break; |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 1234 | } |
| 1235 | } |
| 1236 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1237 | void RSReflectionJava::genTypeCheck(const RSExportType *ET, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1238 | const char *VarName) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1239 | mOut.indent() << "// check " << VarName << "\n"; |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1240 | |
| 1241 | if (ET->getClass() == RSExportType::ExportClassPointer) { |
| 1242 | const RSExportPointerType *EPT = |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1243 | static_cast<const RSExportPointerType *>(ET); |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1244 | ET = EPT->getPointeeType(); |
| 1245 | } |
| 1246 | |
| 1247 | std::string TypeName; |
| 1248 | |
| 1249 | switch (ET->getClass()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1250 | case RSExportType::ExportClassPrimitive: |
| 1251 | case RSExportType::ExportClassVector: |
| 1252 | case RSExportType::ExportClassRecord: { |
| 1253 | TypeName = ET->getElementName(); |
| 1254 | break; |
| 1255 | } |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1256 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1257 | default: |
| 1258 | break; |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1259 | } |
| 1260 | |
| 1261 | if (!TypeName.empty()) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1262 | mOut.indent() << "if (!" << VarName |
| 1263 | << ".getType().getElement().isCompatible(" RS_ELEM_PREFIX |
| 1264 | << TypeName << ")) {\n"; |
| 1265 | mOut.indent() << " throw new RSRuntimeException(\"Type mismatch with " |
| 1266 | << TypeName << "!\");\n"; |
| 1267 | mOut.indent() << "}\n"; |
Stephen Hines | 48b72bf | 2011-06-10 15:37:27 -0700 | [diff] [blame] | 1268 | } |
Stephen Hines | b5a89fb | 2011-05-17 14:48:02 -0700 | [diff] [blame] | 1269 | } |
| 1270 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1271 | void RSReflectionJava::genPrimitiveTypeExportVariable(const RSExportVar *EV) { |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1272 | slangAssert( |
| 1273 | (EV->getType()->getClass() == RSExportType::ExportClassPrimitive) && |
| 1274 | "Variable should be type of primitive here"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1275 | |
| 1276 | const RSExportPrimitiveType *EPT = |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1277 | static_cast<const RSExportPrimitiveType *>(EV->getType()); |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1278 | std::string TypeName = GetTypeName(EPT); |
| 1279 | std::string VarName = EV->getName(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1280 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1281 | genPrivateExportVariable(TypeName, EV->getName()); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1282 | |
Stephen Hines | 5d67178 | 2012-01-31 19:32:04 -0800 | [diff] [blame] | 1283 | if (EV->isConst()) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1284 | mOut.indent() << "public final static " << TypeName |
| 1285 | << " " RS_EXPORT_VAR_CONST_PREFIX << VarName << " = "; |
Stephen Hines | 5d67178 | 2012-01-31 19:32:04 -0800 | [diff] [blame] | 1286 | const clang::APValue &Val = EV->getInit(); |
Jean-Luc Brouillet | efcff10 | 2014-06-03 16:13:51 -0700 | [diff] [blame] | 1287 | genInitValue(Val, EPT->getType() == DataTypeBoolean); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1288 | mOut << ";\n"; |
Stephen Hines | 5d67178 | 2012-01-31 19:32:04 -0800 | [diff] [blame] | 1289 | } else { |
| 1290 | // set_*() |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 1291 | // This must remain synchronized, since multiple Dalvik threads may |
| 1292 | // be calling setters. |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1293 | startFunction(AM_PublicSynchronized, false, "void", "set_" + VarName, 1, |
| 1294 | TypeName.c_str(), "v"); |
Stephen Hines | bcae1fe | 2012-09-25 19:17:33 -0700 | [diff] [blame] | 1295 | if ((EPT->getSize() < 4) || EV->isUnsigned()) { |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 1296 | // We create/cache a per-type FieldPacker. This allows us to reuse the |
| 1297 | // validation logic (for catching negative inputs from Dalvik, as well |
| 1298 | // as inputs that are too large to be represented in the unsigned type). |
Stephen Hines | bcae1fe | 2012-09-25 19:17:33 -0700 | [diff] [blame] | 1299 | // Sub-integer types are also handled specially here, so that we don't |
| 1300 | // overwrite bytes accidentally. |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 1301 | std::string ElemName = EPT->getElementName(); |
| 1302 | std::string FPName; |
| 1303 | FPName = RS_FP_PREFIX + ElemName; |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1304 | mOut.indent() << "if (" << FPName << "!= null) {\n"; |
| 1305 | mOut.increaseIndent(); |
| 1306 | mOut.indent() << FPName << ".reset();\n"; |
| 1307 | mOut.decreaseIndent(); |
| 1308 | mOut.indent() << "} else {\n"; |
| 1309 | mOut.increaseIndent(); |
| 1310 | mOut.indent() << FPName << " = new FieldPacker(" << EPT->getSize() |
| 1311 | << ");\n"; |
| 1312 | mOut.decreaseIndent(); |
| 1313 | mOut.indent() << "}\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1314 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1315 | genPackVarOfType(EPT, "v", FPName.c_str()); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1316 | mOut.indent() << "setVar(" << RS_EXPORT_VAR_INDEX_PREFIX << VarName |
| 1317 | << ", " << FPName << ");\n"; |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 1318 | } else { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1319 | mOut.indent() << "setVar(" << RS_EXPORT_VAR_INDEX_PREFIX << VarName |
| 1320 | << ", v);\n"; |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 1321 | } |
| 1322 | |
| 1323 | // Dalvik update comes last, since the input may be invalid (and hence |
| 1324 | // throw an exception). |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1325 | mOut.indent() << RS_EXPORT_VAR_PREFIX << VarName << " = v;\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1326 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1327 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1328 | } |
Shih-wei Liao | 9e86e19 | 2010-06-18 20:42:28 -0700 | [diff] [blame] | 1329 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1330 | genGetExportVariable(TypeName, VarName); |
| 1331 | genGetFieldID(VarName); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 1332 | } |
| 1333 | |
Jean-Luc Brouillet | efcff10 | 2014-06-03 16:13:51 -0700 | [diff] [blame] | 1334 | void RSReflectionJava::genInitValue(const clang::APValue &Val, bool asBool) { |
| 1335 | switch (Val.getKind()) { |
| 1336 | case clang::APValue::Int: { |
| 1337 | llvm::APInt api = Val.getInt(); |
| 1338 | if (asBool) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1339 | mOut << ((api.getSExtValue() == 0) ? "false" : "true"); |
Jean-Luc Brouillet | efcff10 | 2014-06-03 16:13:51 -0700 | [diff] [blame] | 1340 | } else { |
| 1341 | // TODO: Handle unsigned correctly |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1342 | mOut << api.getSExtValue(); |
Jean-Luc Brouillet | efcff10 | 2014-06-03 16:13:51 -0700 | [diff] [blame] | 1343 | if (api.getBitWidth() > 32) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1344 | mOut << "L"; |
Jean-Luc Brouillet | efcff10 | 2014-06-03 16:13:51 -0700 | [diff] [blame] | 1345 | } |
| 1346 | } |
| 1347 | break; |
| 1348 | } |
| 1349 | |
| 1350 | case clang::APValue::Float: { |
| 1351 | llvm::APFloat apf = Val.getFloat(); |
| 1352 | llvm::SmallString<30> s; |
| 1353 | apf.toString(s); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1354 | mOut << s.c_str(); |
Jean-Luc Brouillet | efcff10 | 2014-06-03 16:13:51 -0700 | [diff] [blame] | 1355 | if (&apf.getSemantics() == &llvm::APFloat::IEEEsingle) { |
| 1356 | if (s.count('.') == 0) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1357 | mOut << ".f"; |
Jean-Luc Brouillet | efcff10 | 2014-06-03 16:13:51 -0700 | [diff] [blame] | 1358 | } else { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1359 | mOut << "f"; |
Jean-Luc Brouillet | efcff10 | 2014-06-03 16:13:51 -0700 | [diff] [blame] | 1360 | } |
| 1361 | } |
| 1362 | break; |
| 1363 | } |
| 1364 | |
| 1365 | case clang::APValue::ComplexInt: |
| 1366 | case clang::APValue::ComplexFloat: |
| 1367 | case clang::APValue::LValue: |
| 1368 | case clang::APValue::Vector: { |
| 1369 | slangAssert(false && "Primitive type cannot have such kind of initializer"); |
| 1370 | break; |
| 1371 | } |
| 1372 | |
| 1373 | default: { slangAssert(false && "Unknown kind of initializer"); } |
| 1374 | } |
| 1375 | } |
| 1376 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1377 | void RSReflectionJava::genPointerTypeExportVariable(const RSExportVar *EV) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1378 | const RSExportType *ET = EV->getType(); |
| 1379 | const RSExportType *PointeeType; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 1380 | |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 1381 | slangAssert((ET->getClass() == RSExportType::ExportClassPointer) && |
| 1382 | "Variable should be type of pointer here"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1383 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1384 | PointeeType = static_cast<const RSExportPointerType *>(ET)->getPointeeType(); |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1385 | std::string TypeName = GetTypeName(ET); |
| 1386 | std::string VarName = EV->getName(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1387 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1388 | genPrivateExportVariable(TypeName, VarName); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1389 | |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 1390 | // bind_*() |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1391 | startFunction(AM_Public, false, "void", "bind_" + VarName, 1, |
| 1392 | TypeName.c_str(), "v"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1393 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1394 | mOut.indent() << RS_EXPORT_VAR_PREFIX << VarName << " = v;\n"; |
| 1395 | mOut.indent() << "if (v == null) bindAllocation(null, " |
| 1396 | << RS_EXPORT_VAR_INDEX_PREFIX << VarName << ");\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1397 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1398 | if (PointeeType->getClass() == RSExportType::ExportClassRecord) { |
| 1399 | mOut.indent() << "else bindAllocation(v.getAllocation(), " |
| 1400 | << RS_EXPORT_VAR_INDEX_PREFIX << VarName << ");\n"; |
| 1401 | } else { |
| 1402 | mOut.indent() << "else bindAllocation(v, " << RS_EXPORT_VAR_INDEX_PREFIX |
| 1403 | << VarName << ");\n"; |
| 1404 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1405 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1406 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1407 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1408 | genGetExportVariable(TypeName, VarName); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1409 | } |
| 1410 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1411 | void RSReflectionJava::genVectorTypeExportVariable(const RSExportVar *EV) { |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 1412 | slangAssert((EV->getType()->getClass() == RSExportType::ExportClassVector) && |
| 1413 | "Variable should be type of vector here"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1414 | |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1415 | std::string TypeName = GetTypeName(EV->getType()); |
| 1416 | std::string VarName = EV->getName(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1417 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1418 | genPrivateExportVariable(TypeName, VarName); |
| 1419 | genSetExportVariable(TypeName, EV); |
| 1420 | genGetExportVariable(TypeName, VarName); |
| 1421 | genGetFieldID(VarName); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1422 | } |
| 1423 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1424 | void RSReflectionJava::genMatrixTypeExportVariable(const RSExportVar *EV) { |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 1425 | slangAssert((EV->getType()->getClass() == RSExportType::ExportClassMatrix) && |
| 1426 | "Variable should be type of matrix here"); |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 1427 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1428 | const RSExportType *ET = EV->getType(); |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1429 | std::string TypeName = GetTypeName(ET); |
| 1430 | std::string VarName = EV->getName(); |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 1431 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1432 | genPrivateExportVariable(TypeName, VarName); |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 1433 | |
| 1434 | // set_*() |
| 1435 | if (!EV->isConst()) { |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1436 | const char *FieldPackerName = "fp"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1437 | startFunction(AM_PublicSynchronized, false, "void", "set_" + VarName, 1, |
| 1438 | TypeName.c_str(), "v"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1439 | mOut.indent() << RS_EXPORT_VAR_PREFIX << VarName << " = v;\n"; |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 1440 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1441 | if (genCreateFieldPacker(ET, FieldPackerName)) |
| 1442 | genPackVarOfType(ET, "v", FieldPackerName); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1443 | mOut.indent() << "setVar(" RS_EXPORT_VAR_INDEX_PREFIX << VarName << ", " |
| 1444 | << FieldPackerName << ");\n"; |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 1445 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1446 | endFunction(); |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 1447 | } |
| 1448 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1449 | genGetExportVariable(TypeName, VarName); |
| 1450 | genGetFieldID(VarName); |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 1451 | } |
| 1452 | |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1453 | void |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1454 | RSReflectionJava::genConstantArrayTypeExportVariable(const RSExportVar *EV) { |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1455 | slangAssert( |
| 1456 | (EV->getType()->getClass() == RSExportType::ExportClassConstantArray) && |
| 1457 | "Variable should be type of constant array here"); |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1458 | |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1459 | std::string TypeName = GetTypeName(EV->getType()); |
| 1460 | std::string VarName = EV->getName(); |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1461 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1462 | genPrivateExportVariable(TypeName, VarName); |
| 1463 | genSetExportVariable(TypeName, EV); |
| 1464 | genGetExportVariable(TypeName, VarName); |
| 1465 | genGetFieldID(VarName); |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1466 | } |
| 1467 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1468 | void RSReflectionJava::genRecordTypeExportVariable(const RSExportVar *EV) { |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 1469 | slangAssert((EV->getType()->getClass() == RSExportType::ExportClassRecord) && |
| 1470 | "Variable should be type of struct here"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1471 | |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1472 | std::string TypeName = GetTypeName(EV->getType()); |
| 1473 | std::string VarName = EV->getName(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1474 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1475 | genPrivateExportVariable(TypeName, VarName); |
| 1476 | genSetExportVariable(TypeName, EV); |
| 1477 | genGetExportVariable(TypeName, VarName); |
| 1478 | genGetFieldID(VarName); |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1479 | } |
| 1480 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1481 | void RSReflectionJava::genPrivateExportVariable(const std::string &TypeName, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1482 | const std::string &VarName) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1483 | mOut.indent() << "private " << TypeName << " " << RS_EXPORT_VAR_PREFIX |
| 1484 | << VarName << ";\n"; |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1485 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1486 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1487 | void RSReflectionJava::genSetExportVariable(const std::string &TypeName, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1488 | const RSExportVar *EV) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1489 | if (!EV->isConst()) { |
Stephen Hines | 0d26cef | 2012-05-01 19:23:01 -0700 | [diff] [blame] | 1490 | const char *FieldPackerName = "fp"; |
| 1491 | std::string VarName = EV->getName(); |
| 1492 | const RSExportType *ET = EV->getType(); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1493 | startFunction(AM_PublicSynchronized, false, "void", "set_" + VarName, 1, |
| 1494 | TypeName.c_str(), "v"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1495 | mOut.indent() << RS_EXPORT_VAR_PREFIX << VarName << " = v;\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1496 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1497 | if (genCreateFieldPacker(ET, FieldPackerName)) |
| 1498 | genPackVarOfType(ET, "v", FieldPackerName); |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 1499 | |
| 1500 | if (mRSContext->getTargetAPI() < SLANG_JB_TARGET_API) { |
| 1501 | // Legacy apps must use the old setVar() without Element/dim components. |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1502 | mOut.indent() << "setVar(" << RS_EXPORT_VAR_INDEX_PREFIX << VarName |
| 1503 | << ", " << FieldPackerName << ");\n"; |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 1504 | } else { |
| 1505 | // We only have support for one-dimensional array reflection today, |
| 1506 | // but the entry point (i.e. setVar()) takes an array of dimensions. |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1507 | mOut.indent() << "int []__dimArr = new int[1];\n"; |
| 1508 | mOut.indent() << "__dimArr[0] = " << ET->getSize() << ";\n"; |
| 1509 | mOut.indent() << "setVar(" << RS_EXPORT_VAR_INDEX_PREFIX << VarName |
| 1510 | << ", " << FieldPackerName << ", " << RS_ELEM_PREFIX |
| 1511 | << ET->getElementName() << ", __dimArr);\n"; |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 1512 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1513 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1514 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1515 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1516 | } |
| 1517 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1518 | void RSReflectionJava::genGetExportVariable(const std::string &TypeName, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1519 | const std::string &VarName) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1520 | startFunction(AM_Public, false, TypeName.c_str(), "get_" + VarName, 0); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1521 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1522 | mOut.indent() << "return " << RS_EXPORT_VAR_PREFIX << VarName << ";\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1523 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1524 | endFunction(); |
Stephen Hines | 28d60bc | 2012-10-15 15:54:16 -0700 | [diff] [blame] | 1525 | } |
| 1526 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1527 | void RSReflectionJava::genGetFieldID(const std::string &VarName) { |
Stephen Hines | 28d60bc | 2012-10-15 15:54:16 -0700 | [diff] [blame] | 1528 | // We only generate getFieldID_*() for non-Pointer (bind) types. |
Tim Murray | b81a993 | 2012-10-10 15:56:02 -0700 | [diff] [blame] | 1529 | if (mRSContext->getTargetAPI() >= SLANG_JB_MR1_TARGET_API) { |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1530 | startFunction(AM_Public, false, "Script.FieldID", "getFieldID_" + VarName, |
| 1531 | 0); |
Tim Murray | b81a993 | 2012-10-10 15:56:02 -0700 | [diff] [blame] | 1532 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1533 | mOut.indent() << "return createFieldID(" << RS_EXPORT_VAR_INDEX_PREFIX |
| 1534 | << VarName << ", null);\n"; |
Tim Murray | b81a993 | 2012-10-10 15:56:02 -0700 | [diff] [blame] | 1535 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1536 | endFunction(); |
Tim Murray | b81a993 | 2012-10-10 15:56:02 -0700 | [diff] [blame] | 1537 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1538 | } |
| 1539 | |
| 1540 | /******************* Methods to generate script class /end *******************/ |
| 1541 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1542 | bool RSReflectionJava::genCreateFieldPacker(const RSExportType *ET, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1543 | const char *FieldPackerName) { |
Jean-Luc Brouillet | c95381a | 2014-05-14 21:24:45 -0700 | [diff] [blame] | 1544 | size_t AllocSize = ET->getAllocSize(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1545 | if (AllocSize > 0) |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1546 | mOut.indent() << "FieldPacker " << FieldPackerName << " = new FieldPacker(" |
| 1547 | << AllocSize << ");\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1548 | else |
| 1549 | return false; |
| 1550 | return true; |
| 1551 | } |
| 1552 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1553 | void RSReflectionJava::genPackVarOfType(const RSExportType *ET, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1554 | const char *VarName, |
| 1555 | const char *FieldPackerName) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1556 | switch (ET->getClass()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1557 | case RSExportType::ExportClassPrimitive: |
| 1558 | case RSExportType::ExportClassVector: { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1559 | mOut.indent() << FieldPackerName << "." |
| 1560 | << GetPackerAPIName( |
| 1561 | static_cast<const RSExportPrimitiveType *>(ET)) << "(" |
| 1562 | << VarName << ");\n"; |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1563 | break; |
| 1564 | } |
| 1565 | case RSExportType::ExportClassPointer: { |
| 1566 | // Must reflect as type Allocation in Java |
| 1567 | const RSExportType *PointeeType = |
| 1568 | static_cast<const RSExportPointerType *>(ET)->getPointeeType(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1569 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1570 | if (PointeeType->getClass() != RSExportType::ExportClassRecord) { |
| 1571 | mOut.indent() << FieldPackerName << ".addI32(" << VarName |
| 1572 | << ".getPtr());\n"; |
| 1573 | } else { |
| 1574 | mOut.indent() << FieldPackerName << ".addI32(" << VarName |
| 1575 | << ".getAllocation().getPtr());\n"; |
| 1576 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1577 | break; |
| 1578 | } |
| 1579 | case RSExportType::ExportClassMatrix: { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1580 | mOut.indent() << FieldPackerName << ".addMatrix(" << VarName << ");\n"; |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1581 | break; |
| 1582 | } |
| 1583 | case RSExportType::ExportClassConstantArray: { |
| 1584 | const RSExportConstantArrayType *ECAT = |
| 1585 | static_cast<const RSExportConstantArrayType *>(ET); |
| 1586 | |
| 1587 | // TODO(zonr): more elegant way. Currently, we obtain the unique index |
| 1588 | // variable (this method involves recursive call which means |
| 1589 | // we may have more than one level loop, therefore we can't |
| 1590 | // always use the same index variable name here) name given |
| 1591 | // in the for-loop from counting the '.' in @VarName. |
| 1592 | unsigned Level = 0; |
| 1593 | size_t LastDotPos = 0; |
| 1594 | std::string ElementVarName(VarName); |
| 1595 | |
| 1596 | while (LastDotPos != std::string::npos) { |
| 1597 | LastDotPos = ElementVarName.find_first_of('.', LastDotPos + 1); |
| 1598 | Level++; |
| 1599 | } |
| 1600 | std::string IndexVarName("ct"); |
| 1601 | IndexVarName.append(llvm::utostr_32(Level)); |
| 1602 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1603 | mOut.indent() << "for (int " << IndexVarName << " = 0; " << IndexVarName |
| 1604 | << " < " << ECAT->getSize() << "; " << IndexVarName << "++)"; |
| 1605 | mOut.startBlock(); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1606 | |
| 1607 | ElementVarName.append("[" + IndexVarName + "]"); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1608 | genPackVarOfType(ECAT->getElementType(), ElementVarName.c_str(), |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1609 | FieldPackerName); |
| 1610 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1611 | mOut.endBlock(); |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1612 | break; |
| 1613 | } |
| 1614 | case RSExportType::ExportClassRecord: { |
| 1615 | const RSExportRecordType *ERT = static_cast<const RSExportRecordType *>(ET); |
| 1616 | // Relative pos from now on in field packer |
| 1617 | unsigned Pos = 0; |
| 1618 | |
| 1619 | for (RSExportRecordType::const_field_iterator I = ERT->fields_begin(), |
| 1620 | E = ERT->fields_end(); |
| 1621 | I != E; I++) { |
| 1622 | const RSExportRecordType::Field *F = *I; |
| 1623 | std::string FieldName; |
| 1624 | size_t FieldOffset = F->getOffsetInParent(); |
| 1625 | const RSExportType *T = F->getType(); |
| 1626 | size_t FieldStoreSize = T->getStoreSize(); |
| 1627 | size_t FieldAllocSize = T->getAllocSize(); |
| 1628 | |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 1629 | if (VarName != nullptr) |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1630 | FieldName = VarName + ("." + F->getName()); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1631 | else |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1632 | FieldName = F->getName(); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 1633 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1634 | if (FieldOffset > Pos) { |
| 1635 | mOut.indent() << FieldPackerName << ".skip(" << (FieldOffset - Pos) |
| 1636 | << ");\n"; |
| 1637 | } |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 1638 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1639 | genPackVarOfType(F->getType(), FieldName.c_str(), FieldPackerName); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 1640 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1641 | // There is padding in the field type |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1642 | if (FieldAllocSize > FieldStoreSize) { |
| 1643 | mOut.indent() << FieldPackerName << ".skip(" |
| 1644 | << (FieldAllocSize - FieldStoreSize) << ");\n"; |
| 1645 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1646 | |
| 1647 | Pos = FieldOffset + FieldAllocSize; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1648 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1649 | |
| 1650 | // There maybe some padding after the struct |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1651 | if (ERT->getAllocSize() > Pos) { |
| 1652 | mOut.indent() << FieldPackerName << ".skip(" << ERT->getAllocSize() - Pos |
| 1653 | << ");\n"; |
| 1654 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1655 | break; |
| 1656 | } |
| 1657 | default: { slangAssert(false && "Unknown class of type"); } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1658 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1659 | } |
| 1660 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1661 | void RSReflectionJava::genAllocateVarOfType(const RSExportType *T, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1662 | const std::string &VarName) { |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1663 | switch (T->getClass()) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1664 | case RSExportType::ExportClassPrimitive: { |
| 1665 | // Primitive type like int in Java has its own storage once it's declared. |
| 1666 | // |
| 1667 | // FIXME: Should we allocate storage for RS object? |
| 1668 | // if (static_cast<const RSExportPrimitiveType *>(T)->isRSObjectType()) |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1669 | // mOut.indent() << VarName << " = new " << GetTypeName(T) << "();\n"; |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1670 | break; |
| 1671 | } |
| 1672 | case RSExportType::ExportClassPointer: { |
| 1673 | // Pointer type is an instance of Allocation or a TypeClass whose value is |
| 1674 | // expected to be assigned by programmer later in Java program. Therefore |
| 1675 | // we don't reflect things like [VarName] = new Allocation(); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1676 | mOut.indent() << VarName << " = null;\n"; |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1677 | break; |
| 1678 | } |
| 1679 | case RSExportType::ExportClassConstantArray: { |
| 1680 | const RSExportConstantArrayType *ECAT = |
| 1681 | static_cast<const RSExportConstantArrayType *>(T); |
| 1682 | const RSExportType *ElementType = ECAT->getElementType(); |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1683 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1684 | mOut.indent() << VarName << " = new " << GetTypeName(ElementType) << "[" |
| 1685 | << ECAT->getSize() << "];\n"; |
Zonr Chang | 2f1451c | 2010-10-14 02:58:28 +0800 | [diff] [blame] | 1686 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1687 | // Primitive type element doesn't need allocation code. |
| 1688 | if (ElementType->getClass() != RSExportType::ExportClassPrimitive) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1689 | mOut.indent() << "for (int $ct = 0; $ct < " << ECAT->getSize() |
| 1690 | << "; $ct++)"; |
| 1691 | mOut.startBlock(); |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1692 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1693 | std::string ElementVarName(VarName); |
| 1694 | ElementVarName.append("[$ct]"); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1695 | genAllocateVarOfType(ElementType, ElementVarName); |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1696 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1697 | mOut.endBlock(); |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1698 | } |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1699 | break; |
| 1700 | } |
| 1701 | case RSExportType::ExportClassVector: |
| 1702 | case RSExportType::ExportClassMatrix: |
| 1703 | case RSExportType::ExportClassRecord: { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1704 | mOut.indent() << VarName << " = new " << GetTypeName(T) << "();\n"; |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1705 | break; |
| 1706 | } |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1707 | } |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 1708 | } |
| 1709 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1710 | void RSReflectionJava::genNewItemBufferIfNull(const char *Index) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1711 | mOut.indent() << "if (" << RS_TYPE_ITEM_BUFFER_NAME " == null) "; |
| 1712 | mOut << RS_TYPE_ITEM_BUFFER_NAME << " = new " << RS_TYPE_ITEM_CLASS_NAME |
| 1713 | << "[getType().getX() /* count */];\n"; |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 1714 | if (Index != nullptr) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1715 | mOut.indent() << "if (" << RS_TYPE_ITEM_BUFFER_NAME << "[" << Index |
| 1716 | << "] == null) "; |
| 1717 | mOut << RS_TYPE_ITEM_BUFFER_NAME << "[" << Index << "] = new " |
| 1718 | << RS_TYPE_ITEM_CLASS_NAME << "();\n"; |
| 1719 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1720 | } |
| 1721 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1722 | void RSReflectionJava::genNewItemBufferPackerIfNull() { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1723 | mOut.indent() << "if (" << RS_TYPE_ITEM_BUFFER_PACKER_NAME << " == null) "; |
| 1724 | mOut << RS_TYPE_ITEM_BUFFER_PACKER_NAME " = new FieldPacker(" |
Tim Murray | 3a38b74 | 2014-07-02 10:41:08 -0700 | [diff] [blame] | 1725 | << mItemSizeof << " * getType().getX()/* count */);\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1726 | } |
| 1727 | |
| 1728 | /********************** Methods to generate type class **********************/ |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1729 | bool RSReflectionJava::genTypeClass(const RSExportRecordType *ERT, |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1730 | std::string &ErrorMsg) { |
Stephen Hines | a6b5414 | 2012-04-09 18:25:08 -0700 | [diff] [blame] | 1731 | std::string ClassName = ERT->getElementName(); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1732 | std::string superClassName = getRSPackageName(); |
Tim Murray | f69e1e5 | 2013-01-17 13:57:24 -0800 | [diff] [blame] | 1733 | superClassName += RS_TYPE_CLASS_SUPER_CLASS_NAME; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1734 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1735 | if (!startClass(AM_Public, false, ClassName, superClassName.c_str(), |
| 1736 | ErrorMsg)) |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1737 | return false; |
| 1738 | |
Stephen Hines | 4cc67fc | 2011-01-31 16:48:57 -0800 | [diff] [blame] | 1739 | mGeneratedFileNames->push_back(ClassName); |
| 1740 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1741 | genTypeItemClass(ERT); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1742 | |
| 1743 | // Declare item buffer and item buffer packer |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1744 | mOut.indent() << "private " << RS_TYPE_ITEM_CLASS_NAME << " " |
| 1745 | << RS_TYPE_ITEM_BUFFER_NAME << "[];\n"; |
| 1746 | mOut.indent() << "private FieldPacker " << RS_TYPE_ITEM_BUFFER_PACKER_NAME |
| 1747 | << ";\n"; |
| 1748 | mOut.indent() << "private static java.lang.ref.WeakReference<Element> " |
| 1749 | << RS_TYPE_ELEMENT_REF_NAME |
| 1750 | << " = new java.lang.ref.WeakReference<Element>(null);\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1751 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1752 | genTypeClassConstructor(ERT); |
| 1753 | genTypeClassCopyToArrayLocal(ERT); |
| 1754 | genTypeClassCopyToArray(ERT); |
| 1755 | genTypeClassItemSetter(ERT); |
| 1756 | genTypeClassItemGetter(ERT); |
| 1757 | genTypeClassComponentSetter(ERT); |
| 1758 | genTypeClassComponentGetter(ERT); |
| 1759 | genTypeClassCopyAll(ERT); |
Stephen Hines | 82754d8 | 2013-01-18 19:46:06 -0800 | [diff] [blame] | 1760 | if (!mRSContext->isCompatLib()) { |
| 1761 | // Skip the resize method if we are targeting a compatibility library. |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1762 | genTypeClassResize(); |
Stephen Hines | 82754d8 | 2013-01-18 19:46:06 -0800 | [diff] [blame] | 1763 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1764 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1765 | endClass(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1766 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1767 | resetFieldIndex(); |
| 1768 | clearFieldIndexMap(); |
Zonr Chang | 66aa299 | 2010-10-05 15:56:31 +0800 | [diff] [blame] | 1769 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1770 | return true; |
| 1771 | } |
| 1772 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1773 | void RSReflectionJava::genTypeItemClass(const RSExportRecordType *ERT) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1774 | mOut.indent() << "static public class " RS_TYPE_ITEM_CLASS_NAME; |
| 1775 | mOut.startBlock(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1776 | |
Tim Murray | 3a38b74 | 2014-07-02 10:41:08 -0700 | [diff] [blame] | 1777 | // Sizeof should not be exposed for 64-bit; it is not accurate |
| 1778 | if (mRSContext->getTargetAPI() < 21) { |
| 1779 | mOut.indent() << "public static final int sizeof = " << ERT->getAllocSize() |
| 1780 | << ";\n"; |
| 1781 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1782 | |
| 1783 | // Member elements |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1784 | mOut << "\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1785 | for (RSExportRecordType::const_field_iterator FI = ERT->fields_begin(), |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1786 | FE = ERT->fields_end(); |
| 1787 | FI != FE; FI++) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1788 | mOut.indent() << GetTypeName((*FI)->getType()) << " " << (*FI)->getName() |
| 1789 | << ";\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1790 | } |
| 1791 | |
| 1792 | // Constructor |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1793 | mOut << "\n"; |
| 1794 | mOut.indent() << RS_TYPE_ITEM_CLASS_NAME << "()"; |
| 1795 | mOut.startBlock(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1796 | |
| 1797 | for (RSExportRecordType::const_field_iterator FI = ERT->fields_begin(), |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 1798 | FE = ERT->fields_end(); |
| 1799 | FI != FE; FI++) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1800 | const RSExportRecordType::Field *F = *FI; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1801 | genAllocateVarOfType(F->getType(), F->getName()); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1802 | } |
| 1803 | |
| 1804 | // end Constructor |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1805 | mOut.endBlock(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1806 | |
| 1807 | // end Item class |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1808 | mOut.endBlock(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1809 | } |
| 1810 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1811 | void RSReflectionJava::genTypeClassConstructor(const RSExportRecordType *ERT) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1812 | const char *RenderScriptVar = "rs"; |
| 1813 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1814 | startFunction(AM_Public, true, "Element", "createElement", 1, "RenderScript", |
| 1815 | RenderScriptVar); |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1816 | |
Stephen Hines | e67239d | 2012-02-24 15:08:36 -0800 | [diff] [blame] | 1817 | // TODO(all): Fix weak-refs + multi-context issue. |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1818 | // mOut.indent() << "Element e = " << RS_TYPE_ELEMENT_REF_NAME |
Jean-Luc Brouillet | 2968921 | 2014-05-27 13:25:31 -0700 | [diff] [blame] | 1819 | // << ".get();\n"; |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1820 | // mOut.indent() << "if (e != null) return e;\n"; |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 1821 | RSReflectionJavaElementBuilder builder("eb", ERT, RenderScriptVar, &mOut, |
| 1822 | mRSContext, this); |
| 1823 | builder.generate(); |
| 1824 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1825 | mOut.indent() << "return eb.create();\n"; |
| 1826 | // mOut.indent() << "e = eb.create();\n"; |
| 1827 | // mOut.indent() << RS_TYPE_ELEMENT_REF_NAME |
Jean-Luc Brouillet | 2968921 | 2014-05-27 13:25:31 -0700 | [diff] [blame] | 1828 | // << " = new java.lang.ref.WeakReference<Element>(e);\n"; |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1829 | // mOut.indent() << "return e;\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1830 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1831 | |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1832 | // private with element |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 1833 | startFunction(AM_Private, false, nullptr, getClassName(), 1, "RenderScript", |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1834 | RenderScriptVar); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1835 | mOut.indent() << RS_TYPE_ITEM_BUFFER_NAME << " = null;\n"; |
| 1836 | mOut.indent() << RS_TYPE_ITEM_BUFFER_PACKER_NAME << " = null;\n"; |
| 1837 | mOut.indent() << "mElement = createElement(" << RenderScriptVar << ");\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1838 | endFunction(); |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1839 | |
| 1840 | // 1D without usage |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 1841 | startFunction(AM_Public, false, nullptr, getClassName(), 2, "RenderScript", |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1842 | RenderScriptVar, "int", "count"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1843 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1844 | mOut.indent() << RS_TYPE_ITEM_BUFFER_NAME << " = null;\n"; |
| 1845 | mOut.indent() << RS_TYPE_ITEM_BUFFER_PACKER_NAME << " = null;\n"; |
| 1846 | mOut.indent() << "mElement = createElement(" << RenderScriptVar << ");\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1847 | // Call init() in super class |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1848 | mOut.indent() << "init(" << RenderScriptVar << ", count);\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1849 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1850 | |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1851 | // 1D with usage |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 1852 | startFunction(AM_Public, false, nullptr, getClassName(), 3, "RenderScript", |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1853 | RenderScriptVar, "int", "count", "int", "usages"); |
Jason Sams | 91fe83b | 2010-12-06 17:07:12 -0800 | [diff] [blame] | 1854 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1855 | mOut.indent() << RS_TYPE_ITEM_BUFFER_NAME << " = null;\n"; |
| 1856 | mOut.indent() << RS_TYPE_ITEM_BUFFER_PACKER_NAME << " = null;\n"; |
| 1857 | mOut.indent() << "mElement = createElement(" << RenderScriptVar << ");\n"; |
Jason Sams | 91fe83b | 2010-12-06 17:07:12 -0800 | [diff] [blame] | 1858 | // Call init() in super class |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1859 | mOut.indent() << "init(" << RenderScriptVar << ", count, usages);\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1860 | endFunction(); |
Jason Sams | 91fe83b | 2010-12-06 17:07:12 -0800 | [diff] [blame] | 1861 | |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1862 | // create1D with usage |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1863 | startFunction(AM_Public, true, getClassName().c_str(), "create1D", 3, |
| 1864 | "RenderScript", RenderScriptVar, "int", "dimX", "int", |
| 1865 | "usages"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1866 | mOut.indent() << getClassName() << " obj = new " << getClassName() << "(" |
| 1867 | << RenderScriptVar << ");\n"; |
| 1868 | mOut.indent() << "obj.mAllocation = Allocation.createSized(" |
| 1869 | "rs, obj.mElement, dimX, usages);\n"; |
| 1870 | mOut.indent() << "return obj;\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1871 | endFunction(); |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1872 | |
| 1873 | // create1D without usage |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1874 | startFunction(AM_Public, true, getClassName().c_str(), "create1D", 2, |
| 1875 | "RenderScript", RenderScriptVar, "int", "dimX"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1876 | mOut.indent() << "return create1D(" << RenderScriptVar |
| 1877 | << ", dimX, Allocation.USAGE_SCRIPT);\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1878 | endFunction(); |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1879 | |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1880 | // create2D without usage |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1881 | startFunction(AM_Public, true, getClassName().c_str(), "create2D", 3, |
| 1882 | "RenderScript", RenderScriptVar, "int", "dimX", "int", "dimY"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1883 | mOut.indent() << "return create2D(" << RenderScriptVar |
| 1884 | << ", dimX, dimY, Allocation.USAGE_SCRIPT);\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1885 | endFunction(); |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1886 | |
| 1887 | // create2D with usage |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1888 | startFunction(AM_Public, true, getClassName().c_str(), "create2D", 4, |
| 1889 | "RenderScript", RenderScriptVar, "int", "dimX", "int", "dimY", |
| 1890 | "int", "usages"); |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1891 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1892 | mOut.indent() << getClassName() << " obj = new " << getClassName() << "(" |
| 1893 | << RenderScriptVar << ");\n"; |
| 1894 | mOut.indent() << "Type.Builder b = new Type.Builder(rs, obj.mElement);\n"; |
| 1895 | mOut.indent() << "b.setX(dimX);\n"; |
| 1896 | mOut.indent() << "b.setY(dimY);\n"; |
| 1897 | mOut.indent() << "Type t = b.create();\n"; |
| 1898 | mOut.indent() << "obj.mAllocation = Allocation.createTyped(rs, t, usages);\n"; |
| 1899 | mOut.indent() << "return obj;\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1900 | endFunction(); |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1901 | |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1902 | // createTypeBuilder |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1903 | startFunction(AM_Public, true, "Type.Builder", "createTypeBuilder", 1, |
| 1904 | "RenderScript", RenderScriptVar); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1905 | mOut.indent() << "Element e = createElement(" << RenderScriptVar << ");\n"; |
| 1906 | mOut.indent() << "return new Type.Builder(rs, e);\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1907 | endFunction(); |
Jason Sams | 381e95f | 2011-11-30 14:01:43 -0800 | [diff] [blame] | 1908 | |
| 1909 | // createCustom with usage |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1910 | startFunction(AM_Public, true, getClassName().c_str(), "createCustom", 3, |
| 1911 | "RenderScript", RenderScriptVar, "Type.Builder", "tb", "int", |
| 1912 | "usages"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1913 | mOut.indent() << getClassName() << " obj = new " << getClassName() << "(" |
| 1914 | << RenderScriptVar << ");\n"; |
| 1915 | mOut.indent() << "Type t = tb.create();\n"; |
| 1916 | mOut.indent() << "if (t.getElement() != obj.mElement) {\n"; |
| 1917 | mOut.indent() << " throw new RSIllegalArgumentException(" |
| 1918 | "\"Type.Builder did not match expected element type.\");\n"; |
| 1919 | mOut.indent() << "}\n"; |
| 1920 | mOut.indent() << "obj.mAllocation = Allocation.createTyped(rs, t, usages);\n"; |
| 1921 | mOut.indent() << "return obj;\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1922 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1923 | } |
| 1924 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1925 | void RSReflectionJava::genTypeClassCopyToArray(const RSExportRecordType *ERT) { |
| 1926 | startFunction(AM_Private, false, "void", "copyToArray", 2, |
| 1927 | RS_TYPE_ITEM_CLASS_NAME, "i", "int", "index"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1928 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1929 | genNewItemBufferPackerIfNull(); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1930 | mOut.indent() << RS_TYPE_ITEM_BUFFER_PACKER_NAME << ".reset(index * " |
Tim Murray | 3a38b74 | 2014-07-02 10:41:08 -0700 | [diff] [blame] | 1931 | << mItemSizeof << ");\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1932 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1933 | mOut.indent() << "copyToArrayLocal(i, " RS_TYPE_ITEM_BUFFER_PACKER_NAME |
| 1934 | ");\n"; |
Alex Sakhartchouk | 38eca1a | 2011-08-25 10:47:52 -0700 | [diff] [blame] | 1935 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1936 | endFunction(); |
Alex Sakhartchouk | 38eca1a | 2011-08-25 10:47:52 -0700 | [diff] [blame] | 1937 | } |
| 1938 | |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1939 | void |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1940 | RSReflectionJava::genTypeClassCopyToArrayLocal(const RSExportRecordType *ERT) { |
| 1941 | startFunction(AM_Private, false, "void", "copyToArrayLocal", 2, |
| 1942 | RS_TYPE_ITEM_CLASS_NAME, "i", "FieldPacker", "fp"); |
Alex Sakhartchouk | 38eca1a | 2011-08-25 10:47:52 -0700 | [diff] [blame] | 1943 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1944 | genPackVarOfType(ERT, "i", "fp"); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1945 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1946 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1947 | } |
| 1948 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1949 | void RSReflectionJava::genTypeClassItemSetter(const RSExportRecordType *ERT) { |
| 1950 | startFunction(AM_PublicSynchronized, false, "void", "set", 3, |
| 1951 | RS_TYPE_ITEM_CLASS_NAME, "i", "int", "index", "boolean", |
| 1952 | "copyNow"); |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 1953 | genNewItemBufferIfNull(nullptr); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1954 | mOut.indent() << RS_TYPE_ITEM_BUFFER_NAME << "[index] = i;\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1955 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1956 | mOut.indent() << "if (copyNow) "; |
| 1957 | mOut.startBlock(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1958 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1959 | mOut.indent() << "copyToArray(i, index);\n"; |
Tim Murray | 3a38b74 | 2014-07-02 10:41:08 -0700 | [diff] [blame] | 1960 | mOut.indent() << "FieldPacker fp = new FieldPacker(" << mItemSizeof << ");\n"; |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1961 | mOut.indent() << "copyToArrayLocal(i, fp);\n"; |
| 1962 | mOut.indent() << "mAllocation.setFromFieldPacker(index, fp);\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1963 | |
| 1964 | // End of if (copyNow) |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1965 | mOut.endBlock(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1966 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1967 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1968 | } |
| 1969 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1970 | void RSReflectionJava::genTypeClassItemGetter(const RSExportRecordType *ERT) { |
| 1971 | startFunction(AM_PublicSynchronized, false, RS_TYPE_ITEM_CLASS_NAME, "get", 1, |
| 1972 | "int", "index"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1973 | mOut.indent() << "if (" << RS_TYPE_ITEM_BUFFER_NAME |
| 1974 | << " == null) return null;\n"; |
| 1975 | mOut.indent() << "return " << RS_TYPE_ITEM_BUFFER_NAME << "[index];\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1976 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1977 | } |
| 1978 | |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1979 | void |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1980 | RSReflectionJava::genTypeClassComponentSetter(const RSExportRecordType *ERT) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1981 | for (RSExportRecordType::const_field_iterator FI = ERT->fields_begin(), |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 1982 | FE = ERT->fields_end(); |
| 1983 | FI != FE; FI++) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1984 | const RSExportRecordType::Field *F = *FI; |
| 1985 | size_t FieldOffset = F->getOffsetInParent(); |
Jean-Luc Brouillet | c95381a | 2014-05-14 21:24:45 -0700 | [diff] [blame] | 1986 | size_t FieldStoreSize = F->getType()->getStoreSize(); |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1987 | unsigned FieldIndex = getFieldIndex(F); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 1988 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 1989 | startFunction(AM_PublicSynchronized, false, "void", "set_" + F->getName(), |
| 1990 | 3, "int", "index", GetTypeName(F->getType()).c_str(), "v", |
| 1991 | "boolean", "copyNow"); |
| 1992 | genNewItemBufferPackerIfNull(); |
| 1993 | genNewItemBufferIfNull("index"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1994 | mOut.indent() << RS_TYPE_ITEM_BUFFER_NAME << "[index]." << F->getName() |
| 1995 | << " = v;\n"; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 1996 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 1997 | mOut.indent() << "if (copyNow) "; |
| 1998 | mOut.startBlock(); |
Shih-wei Liao | 2dd42ff | 2010-06-15 00:34:38 -0700 | [diff] [blame] | 1999 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2000 | if (FieldOffset > 0) { |
| 2001 | mOut.indent() << RS_TYPE_ITEM_BUFFER_PACKER_NAME << ".reset(index * " |
Tim Murray | 3a38b74 | 2014-07-02 10:41:08 -0700 | [diff] [blame] | 2002 | << mItemSizeof << " + " << FieldOffset |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2003 | << ");\n"; |
| 2004 | } else { |
| 2005 | mOut.indent() << RS_TYPE_ITEM_BUFFER_PACKER_NAME << ".reset(index * " |
Tim Murray | 3a38b74 | 2014-07-02 10:41:08 -0700 | [diff] [blame] | 2006 | << mItemSizeof << ");\n"; |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2007 | } |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2008 | genPackVarOfType(F->getType(), "v", RS_TYPE_ITEM_BUFFER_PACKER_NAME); |
| 2009 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2010 | mOut.indent() << "FieldPacker fp = new FieldPacker(" << FieldStoreSize |
| 2011 | << ");\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2012 | genPackVarOfType(F->getType(), "v", "fp"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2013 | mOut.indent() << "mAllocation.setFromFieldPacker(index, " << FieldIndex |
| 2014 | << ", fp);\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2015 | |
| 2016 | // End of if (copyNow) |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2017 | mOut.endBlock(); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2018 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2019 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2020 | } |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2021 | } |
| 2022 | |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 2023 | void |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2024 | RSReflectionJava::genTypeClassComponentGetter(const RSExportRecordType *ERT) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2025 | for (RSExportRecordType::const_field_iterator FI = ERT->fields_begin(), |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 2026 | FE = ERT->fields_end(); |
| 2027 | FI != FE; FI++) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2028 | const RSExportRecordType::Field *F = *FI; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2029 | startFunction(AM_PublicSynchronized, false, |
| 2030 | GetTypeName(F->getType()).c_str(), "get_" + F->getName(), 1, |
| 2031 | "int", "index"); |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2032 | mOut.indent() << "if (" RS_TYPE_ITEM_BUFFER_NAME << " == null) return " |
| 2033 | << GetTypeNullValue(F->getType()) << ";\n"; |
| 2034 | mOut.indent() << "return " RS_TYPE_ITEM_BUFFER_NAME << "[index]." |
| 2035 | << F->getName() << ";\n"; |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2036 | endFunction(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2037 | } |
Shih-wei Liao | 9c631ff | 2010-09-17 11:57:29 -0700 | [diff] [blame] | 2038 | } |
| 2039 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2040 | void RSReflectionJava::genTypeClassCopyAll(const RSExportRecordType *ERT) { |
| 2041 | startFunction(AM_PublicSynchronized, false, "void", "copyAll", 0); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2042 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2043 | mOut.indent() << "for (int ct = 0; ct < " << RS_TYPE_ITEM_BUFFER_NAME |
| 2044 | << ".length; ct++)" |
| 2045 | << " copyToArray(" << RS_TYPE_ITEM_BUFFER_NAME |
| 2046 | << "[ct], ct);\n"; |
| 2047 | mOut.indent() << "mAllocation.setFromFieldPacker(0, " |
| 2048 | << RS_TYPE_ITEM_BUFFER_PACKER_NAME ");\n"; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2049 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2050 | endFunction(); |
Shih-wei Liao | 9c631ff | 2010-09-17 11:57:29 -0700 | [diff] [blame] | 2051 | } |
| 2052 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2053 | void RSReflectionJava::genTypeClassResize() { |
| 2054 | startFunction(AM_PublicSynchronized, false, "void", "resize", 1, "int", |
| 2055 | "newSize"); |
Zonr Chang | d42a429 | 2010-10-17 02:38:43 +0800 | [diff] [blame] | 2056 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2057 | mOut.indent() << "if (mItemArray != null) "; |
| 2058 | mOut.startBlock(); |
| 2059 | mOut.indent() << "int oldSize = mItemArray.length;\n"; |
| 2060 | mOut.indent() << "int copySize = Math.min(oldSize, newSize);\n"; |
| 2061 | mOut.indent() << "if (newSize == oldSize) return;\n"; |
| 2062 | mOut.indent() << "Item ni[] = new Item[newSize];\n"; |
| 2063 | mOut.indent() << "System.arraycopy(mItemArray, 0, ni, 0, copySize);\n"; |
| 2064 | mOut.indent() << "mItemArray = ni;\n"; |
| 2065 | mOut.endBlock(); |
| 2066 | mOut.indent() << "mAllocation.resize(newSize);\n"; |
Zonr Chang | d42a429 | 2010-10-17 02:38:43 +0800 | [diff] [blame] | 2067 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2068 | mOut.indent() << "if (" RS_TYPE_ITEM_BUFFER_PACKER_NAME |
| 2069 | " != null) " RS_TYPE_ITEM_BUFFER_PACKER_NAME " = " |
Tim Murray | 3a38b74 | 2014-07-02 10:41:08 -0700 | [diff] [blame] | 2070 | "new FieldPacker(" << mItemSizeof << " * getType().getX()/* count */);\n"; |
Jason Sams | cedffd9 | 2010-12-16 15:29:49 -0800 | [diff] [blame] | 2071 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2072 | endFunction(); |
Zonr Chang | d42a429 | 2010-10-17 02:38:43 +0800 | [diff] [blame] | 2073 | } |
| 2074 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2075 | /******************** Methods to generate type class /end ********************/ |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2076 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2077 | /********** Methods to create Element in Java of given record type ***********/ |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2078 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2079 | RSReflectionJavaElementBuilder::RSReflectionJavaElementBuilder( |
| 2080 | const char *ElementBuilderName, const RSExportRecordType *ERT, |
| 2081 | const char *RenderScriptVar, GeneratedFile *Out, const RSContext *RSContext, |
| 2082 | RSReflectionJava *Reflection) |
| 2083 | : mElementBuilderName(ElementBuilderName), mERT(ERT), |
| 2084 | mRenderScriptVar(RenderScriptVar), mOut(Out), mPaddingFieldIndex(1), |
| 2085 | mRSContext(RSContext), mReflection(Reflection) { |
| 2086 | if (mRSContext->getTargetAPI() < SLANG_ICS_TARGET_API) { |
| 2087 | mPaddingPrefix = "#padding_"; |
| 2088 | } else { |
| 2089 | mPaddingPrefix = "#rs_padding_"; |
| 2090 | } |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2091 | } |
| 2092 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2093 | void RSReflectionJavaElementBuilder::generate() { |
| 2094 | mOut->indent() << "Element.Builder " << mElementBuilderName |
| 2095 | << " = new Element.Builder(" << mRenderScriptVar << ");\n"; |
| 2096 | genAddElement(mERT, "", /* ArraySize = */ 0); |
| 2097 | } |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2098 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2099 | void RSReflectionJavaElementBuilder::genAddElement(const RSExportType *ET, |
| 2100 | const std::string &VarName, |
| 2101 | unsigned ArraySize) { |
Stephen Hines | 47aca4e | 2012-03-08 20:07:28 -0800 | [diff] [blame] | 2102 | std::string ElementConstruct = GetBuiltinElementConstruct(ET); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2103 | |
Stephen Hines | 47aca4e | 2012-03-08 20:07:28 -0800 | [diff] [blame] | 2104 | if (ElementConstruct != "") { |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2105 | genAddStatementStart(); |
| 2106 | *mOut << ElementConstruct << "(" << mRenderScriptVar << ")"; |
| 2107 | genAddStatementEnd(VarName, ArraySize); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2108 | } else { |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2109 | |
| 2110 | switch (ET->getClass()) { |
| 2111 | case RSExportType::ExportClassPrimitive: { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2112 | const RSExportPrimitiveType *EPT = |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 2113 | static_cast<const RSExportPrimitiveType *>(ET); |
Stephen Hines | 47aca4e | 2012-03-08 20:07:28 -0800 | [diff] [blame] | 2114 | const char *DataTypeName = |
| 2115 | RSExportPrimitiveType::getRSReflectionType(EPT)->rs_type; |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2116 | genAddStatementStart(); |
| 2117 | *mOut << "Element.createUser(" << mRenderScriptVar |
| 2118 | << ", Element.DataType." << DataTypeName << ")"; |
| 2119 | genAddStatementEnd(VarName, ArraySize); |
| 2120 | break; |
| 2121 | } |
| 2122 | case RSExportType::ExportClassVector: { |
| 2123 | const RSExportVectorType *EVT = |
| 2124 | static_cast<const RSExportVectorType *>(ET); |
| 2125 | const char *DataTypeName = |
| 2126 | RSExportPrimitiveType::getRSReflectionType(EVT)->rs_type; |
| 2127 | genAddStatementStart(); |
| 2128 | *mOut << "Element.createVector(" << mRenderScriptVar |
| 2129 | << ", Element.DataType." << DataTypeName << ", " |
| 2130 | << EVT->getNumElement() << ")"; |
| 2131 | genAddStatementEnd(VarName, ArraySize); |
| 2132 | break; |
| 2133 | } |
| 2134 | case RSExportType::ExportClassPointer: |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2135 | // Pointer type variable should be resolved in |
| 2136 | // GetBuiltinElementConstruct() |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 2137 | slangAssert(false && "??"); |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2138 | break; |
| 2139 | case RSExportType::ExportClassMatrix: |
Zonr Chang | 92b344a | 2010-10-05 20:39:03 +0800 | [diff] [blame] | 2140 | // Matrix type variable should be resolved |
| 2141 | // in GetBuiltinElementConstruct() |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 2142 | slangAssert(false && "??"); |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2143 | break; |
| 2144 | case RSExportType::ExportClassConstantArray: { |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 2145 | const RSExportConstantArrayType *ECAT = |
| 2146 | static_cast<const RSExportConstantArrayType *>(ET); |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 2147 | |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2148 | const RSExportType *ElementType = ECAT->getElementType(); |
| 2149 | if (ElementType->getClass() != RSExportType::ExportClassRecord) { |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2150 | genAddElement(ECAT->getElementType(), VarName, ECAT->getSize()); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2151 | } else { |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2152 | std::string NewElementBuilderName(mElementBuilderName); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2153 | NewElementBuilderName.append(1, '_'); |
Zonr Chang | 2e1dba6 | 2010-10-05 22:20:11 +0800 | [diff] [blame] | 2154 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2155 | RSReflectionJavaElementBuilder builder( |
| 2156 | NewElementBuilderName.c_str(), |
| 2157 | static_cast<const RSExportRecordType *>(ElementType), |
| 2158 | mRenderScriptVar, mOut, mRSContext, mReflection); |
| 2159 | builder.generate(); |
| 2160 | |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2161 | ArraySize = ECAT->getSize(); |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2162 | genAddStatementStart(); |
| 2163 | *mOut << NewElementBuilderName << ".create()"; |
| 2164 | genAddStatementEnd(VarName, ArraySize); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2165 | } |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2166 | break; |
| 2167 | } |
| 2168 | case RSExportType::ExportClassRecord: { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2169 | // Simalar to case of RSExportType::ExportClassRecord in genPackVarOfType. |
| 2170 | // |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 2171 | // TODO(zonr): Generalize these two function such that there's no |
| 2172 | // duplicated codes. |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2173 | const RSExportRecordType *ERT = |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 2174 | static_cast<const RSExportRecordType *>(ET); |
| 2175 | int Pos = 0; // relative pos from now on |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2176 | |
| 2177 | for (RSExportRecordType::const_field_iterator I = ERT->fields_begin(), |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 2178 | E = ERT->fields_end(); |
| 2179 | I != E; I++) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2180 | const RSExportRecordType::Field *F = *I; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2181 | int FieldOffset = F->getOffsetInParent(); |
Jean-Luc Brouillet | c95381a | 2014-05-14 21:24:45 -0700 | [diff] [blame] | 2182 | const RSExportType *T = F->getType(); |
| 2183 | int FieldStoreSize = T->getStoreSize(); |
| 2184 | int FieldAllocSize = T->getAllocSize(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2185 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2186 | std::string FieldName; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2187 | if (!VarName.empty()) |
| 2188 | FieldName = VarName + "." + F->getName(); |
| 2189 | else |
| 2190 | FieldName = F->getName(); |
| 2191 | |
| 2192 | // Alignment |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2193 | genAddPadding(FieldOffset - Pos); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2194 | |
| 2195 | // eb.add(...) |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2196 | mReflection->addFieldIndexMapping(F); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2197 | if (F->getType()->getClass() != RSExportType::ExportClassRecord) { |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2198 | genAddElement(F->getType(), FieldName, 0); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2199 | } else { |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2200 | std::string NewElementBuilderName(mElementBuilderName); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2201 | NewElementBuilderName.append(1, '_'); |
| 2202 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2203 | RSReflectionJavaElementBuilder builder( |
| 2204 | NewElementBuilderName.c_str(), |
| 2205 | static_cast<const RSExportRecordType *>(F->getType()), |
| 2206 | mRenderScriptVar, mOut, mRSContext, mReflection); |
| 2207 | builder.generate(); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2208 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2209 | genAddStatementStart(); |
| 2210 | *mOut << NewElementBuilderName << ".create()"; |
| 2211 | genAddStatementEnd(FieldName, ArraySize); |
Zonr Chang | 89273bd | 2010-10-14 20:57:38 +0800 | [diff] [blame] | 2212 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2213 | |
Stephen Hines | a9ae5ae | 2011-11-11 21:16:59 -0800 | [diff] [blame] | 2214 | if (mRSContext->getTargetAPI() < SLANG_ICS_TARGET_API) { |
| 2215 | // There is padding within the field type. This is only necessary |
| 2216 | // for HC-targeted APIs. |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2217 | genAddPadding(FieldAllocSize - FieldStoreSize); |
Stephen Hines | a9ae5ae | 2011-11-11 21:16:59 -0800 | [diff] [blame] | 2218 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2219 | |
| 2220 | Pos = FieldOffset + FieldAllocSize; |
| 2221 | } |
| 2222 | |
| 2223 | // There maybe some padding after the struct |
Jean-Luc Brouillet | c95381a | 2014-05-14 21:24:45 -0700 | [diff] [blame] | 2224 | size_t RecordAllocSize = ERT->getAllocSize(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2225 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2226 | genAddPadding(RecordAllocSize - Pos); |
| 2227 | break; |
| 2228 | } |
| 2229 | default: |
Stephen Hines | 6e6578a | 2011-02-07 18:05:48 -0800 | [diff] [blame] | 2230 | slangAssert(false && "Unknown class of type"); |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2231 | break; |
Shih-wei Liao | b1a28e7 | 2010-06-16 16:31:32 -0700 | [diff] [blame] | 2232 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2233 | } |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2234 | } |
| 2235 | |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2236 | void RSReflectionJavaElementBuilder::genAddPadding(int PaddingSize) { |
| 2237 | while (PaddingSize > 0) { |
| 2238 | const std::string &VarName = createPaddingField(); |
| 2239 | genAddStatementStart(); |
| 2240 | if (PaddingSize >= 4) { |
| 2241 | *mOut << "Element.U32(" << mRenderScriptVar << ")"; |
| 2242 | PaddingSize -= 4; |
| 2243 | } else if (PaddingSize >= 2) { |
| 2244 | *mOut << "Element.U16(" << mRenderScriptVar << ")"; |
| 2245 | PaddingSize -= 2; |
| 2246 | } else if (PaddingSize >= 1) { |
| 2247 | *mOut << "Element.U8(" << mRenderScriptVar << ")"; |
| 2248 | PaddingSize -= 1; |
| 2249 | } |
| 2250 | genAddStatementEnd(VarName, 0); |
| 2251 | } |
| 2252 | } |
| 2253 | |
| 2254 | void RSReflectionJavaElementBuilder::genAddStatementStart() { |
| 2255 | mOut->indent() << mElementBuilderName << ".add("; |
| 2256 | } |
| 2257 | |
Jean-Luc Brouillet | 602def7 | 2014-05-27 16:11:37 -0700 | [diff] [blame] | 2258 | void |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2259 | RSReflectionJavaElementBuilder::genAddStatementEnd(const std::string &VarName, |
| 2260 | unsigned ArraySize) { |
| 2261 | *mOut << ", \"" << VarName << "\""; |
| 2262 | if (ArraySize > 0) { |
| 2263 | *mOut << ", " << ArraySize; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2264 | } |
Jean-Luc Brouillet | c643ceb | 2014-06-05 10:17:23 -0700 | [diff] [blame] | 2265 | *mOut << ");\n"; |
| 2266 | // TODO Review incFieldIndex. It's probably better to assign the numbers at |
| 2267 | // the start rather |
| 2268 | // than as we're generating the code. |
| 2269 | mReflection->incFieldIndex(); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2270 | } |
| 2271 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2272 | /******** Methods to create Element in Java of given record type /end ********/ |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2273 | |
Jean-Luc Brouillet | 59f22c3 | 2014-06-04 14:53:48 -0700 | [diff] [blame] | 2274 | bool RSReflectionJava::reflect() { |
| 2275 | std::string ErrorMsg; |
| 2276 | if (!genScriptClass(mScriptClassName, ErrorMsg)) { |
| 2277 | std::cerr << "Failed to generate class " << mScriptClassName << " (" |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2278 | << ErrorMsg << ")\n"; |
| 2279 | return false; |
| 2280 | } |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2281 | |
Jean-Luc Brouillet | 59f22c3 | 2014-06-04 14:53:48 -0700 | [diff] [blame] | 2282 | mGeneratedFileNames->push_back(mScriptClassName); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2283 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2284 | // class ScriptField_<TypeName> |
| 2285 | for (RSContext::const_export_type_iterator |
| 2286 | TI = mRSContext->export_types_begin(), |
| 2287 | TE = mRSContext->export_types_end(); |
| 2288 | TI != TE; TI++) { |
| 2289 | const RSExportType *ET = TI->getValue(); |
| 2290 | |
| 2291 | if (ET->getClass() == RSExportType::ExportClassRecord) { |
| 2292 | const RSExportRecordType *ERT = |
| 2293 | static_cast<const RSExportRecordType *>(ET); |
| 2294 | |
| 2295 | if (!ERT->isArtificial() && !genTypeClass(ERT, ErrorMsg)) { |
| 2296 | std::cerr << "Failed to generate type class for struct '" |
| 2297 | << ERT->getName() << "' (" << ErrorMsg << ")\n"; |
| 2298 | return false; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2299 | } |
| 2300 | } |
| 2301 | } |
| 2302 | |
| 2303 | return true; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2304 | } |
| 2305 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2306 | const char *RSReflectionJava::AccessModifierStr(AccessModifier AM) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2307 | switch (AM) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 2308 | case AM_Public: |
| 2309 | return "public"; |
| 2310 | break; |
| 2311 | case AM_Protected: |
| 2312 | return "protected"; |
| 2313 | break; |
| 2314 | case AM_Private: |
| 2315 | return "private"; |
| 2316 | break; |
| 2317 | case AM_PublicSynchronized: |
| 2318 | return "public synchronized"; |
| 2319 | break; |
| 2320 | default: |
| 2321 | return ""; |
| 2322 | break; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2323 | } |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2324 | } |
| 2325 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2326 | bool RSReflectionJava::startClass(AccessModifier AM, bool IsStatic, |
| 2327 | const std::string &ClassName, |
| 2328 | const char *SuperClassName, |
| 2329 | std::string &ErrorMsg) { |
Zonr Chang | 8c6d9b2 | 2010-10-07 18:01:19 +0800 | [diff] [blame] | 2330 | // Open file for class |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2331 | std::string FileName = ClassName + ".java"; |
Jean-Luc Brouillet | 59f22c3 | 2014-06-04 14:53:48 -0700 | [diff] [blame] | 2332 | if (!mOut.startFile(mOutputDirectory, FileName, mRSSourceFileName, |
Stephen Hines | fc4f78b | 2014-06-10 18:07:10 -0700 | [diff] [blame] | 2333 | mRSContext->getLicenseNote(), true, |
| 2334 | mRSContext->getVerbose())) { |
Zonr Chang | 8c6d9b2 | 2010-10-07 18:01:19 +0800 | [diff] [blame] | 2335 | return false; |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2336 | } |
Ying Wang | 4e34844 | 2010-08-18 10:29:12 -0700 | [diff] [blame] | 2337 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2338 | // Package |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2339 | if (!mPackageName.empty()) { |
| 2340 | mOut << "package " << mPackageName << ";\n"; |
| 2341 | } |
| 2342 | mOut << "\n"; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2343 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2344 | // Imports |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2345 | mOut << "import " << mRSPackageName << ".*;\n"; |
Stephen Hines | 44d495d | 2014-05-22 19:42:55 -0700 | [diff] [blame] | 2346 | if (getEmbedBitcodeInJava()) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2347 | mOut << "import " << mPackageName << "." |
Stephen Hines | 44d495d | 2014-05-22 19:42:55 -0700 | [diff] [blame] | 2348 | << RSSlangReflectUtils::JavaBitcodeClassNameFromRSFileName( |
Jean-Luc Brouillet | 59f22c3 | 2014-06-04 14:53:48 -0700 | [diff] [blame] | 2349 | mRSSourceFileName.c_str()) << ";\n"; |
Stephen Hines | 44d495d | 2014-05-22 19:42:55 -0700 | [diff] [blame] | 2350 | } else { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2351 | mOut << "import android.content.res.Resources;\n"; |
Stephen Hines | 44d495d | 2014-05-22 19:42:55 -0700 | [diff] [blame] | 2352 | } |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2353 | mOut << "\n"; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2354 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2355 | // All reflected classes should be annotated as hidden, so that they won't |
| 2356 | // be exposed in SDK. |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2357 | mOut << "/**\n"; |
| 2358 | mOut << " * @hide\n"; |
| 2359 | mOut << " */\n"; |
Ying Wang | 4e34844 | 2010-08-18 10:29:12 -0700 | [diff] [blame] | 2360 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2361 | mOut << AccessModifierStr(AM) << ((IsStatic) ? " static" : "") << " class " |
| 2362 | << ClassName; |
Chris Wailes | 5abbe0e | 2014-08-12 15:58:29 -0700 | [diff] [blame] | 2363 | if (SuperClassName != nullptr) |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2364 | mOut << " extends " << SuperClassName; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2365 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2366 | mOut.startBlock(); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2367 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2368 | mClassName = ClassName; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2369 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2370 | return true; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2371 | } |
| 2372 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2373 | void RSReflectionJava::endClass() { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2374 | mOut.endBlock(); |
| 2375 | mOut.closeFile(); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2376 | clear(); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2377 | } |
| 2378 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2379 | void RSReflectionJava::startTypeClass(const std::string &ClassName) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2380 | mOut.indent() << "public static class " << ClassName; |
| 2381 | mOut.startBlock(); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2382 | } |
| 2383 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2384 | void RSReflectionJava::endTypeClass() { mOut.endBlock(); } |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2385 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2386 | void RSReflectionJava::startFunction(AccessModifier AM, bool IsStatic, |
| 2387 | const char *ReturnType, |
| 2388 | const std::string &FunctionName, int Argc, |
| 2389 | ...) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2390 | ArgTy Args; |
| 2391 | va_list vl; |
| 2392 | va_start(vl, Argc); |
| 2393 | |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 2394 | for (int i = 0; i < Argc; i++) { |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 2395 | const char *ArgType = va_arg(vl, const char *); |
| 2396 | const char *ArgName = va_arg(vl, const char *); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2397 | |
zonr | 6315f76 | 2010-10-05 15:35:14 +0800 | [diff] [blame] | 2398 | Args.push_back(std::make_pair(ArgType, ArgName)); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2399 | } |
| 2400 | va_end(vl); |
| 2401 | |
| 2402 | startFunction(AM, IsStatic, ReturnType, FunctionName, Args); |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2403 | } |
| 2404 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2405 | void RSReflectionJava::startFunction(AccessModifier AM, bool IsStatic, |
| 2406 | const char *ReturnType, |
| 2407 | const std::string &FunctionName, |
| 2408 | const ArgTy &Args) { |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2409 | mOut.indent() << AccessModifierStr(AM) << ((IsStatic) ? " static " : " ") |
| 2410 | << ((ReturnType) ? ReturnType : "") << " " << FunctionName |
| 2411 | << "("; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2412 | |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2413 | bool FirstArg = true; |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 2414 | for (ArgTy::const_iterator I = Args.begin(), E = Args.end(); I != E; I++) { |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2415 | if (!FirstArg) |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2416 | mOut << ", "; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2417 | else |
| 2418 | FirstArg = false; |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2419 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2420 | mOut << I->first << " " << I->second; |
Shih-wei Liao | 9ef2f78 | 2010-10-01 12:31:37 -0700 | [diff] [blame] | 2421 | } |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2422 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2423 | mOut << ")"; |
| 2424 | mOut.startBlock(); |
Shih-wei Liao | 462aefd | 2010-06-04 15:32:04 -0700 | [diff] [blame] | 2425 | } |
| 2426 | |
Jean-Luc Brouillet | f33e156 | 2014-06-03 17:55:57 -0700 | [diff] [blame] | 2427 | void RSReflectionJava::endFunction() { mOut.endBlock(); } |
Stephen Hines | e639eb5 | 2010-11-08 19:27:20 -0800 | [diff] [blame] | 2428 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2429 | bool RSReflectionJava::addTypeNameForElement(const std::string &TypeName) { |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 2430 | if (mTypesToCheck.find(TypeName) == mTypesToCheck.end()) { |
| 2431 | mTypesToCheck.insert(TypeName); |
| 2432 | return true; |
| 2433 | } else { |
| 2434 | return false; |
| 2435 | } |
| 2436 | } |
| 2437 | |
Jean-Luc Brouillet | 2e205d0 | 2014-06-02 21:06:52 -0700 | [diff] [blame] | 2438 | bool RSReflectionJava::addTypeNameForFieldPacker(const std::string &TypeName) { |
Stephen Hines | 1f6c331 | 2012-07-03 17:23:33 -0700 | [diff] [blame] | 2439 | if (mFieldPackerTypes.find(TypeName) == mFieldPackerTypes.end()) { |
| 2440 | mFieldPackerTypes.insert(TypeName); |
| 2441 | return true; |
| 2442 | } else { |
| 2443 | return false; |
| 2444 | } |
| 2445 | } |
| 2446 | |
Jean-Luc Brouillet | 2ce118e | 2014-05-27 17:41:22 -0700 | [diff] [blame] | 2447 | } // namespace slang |