1 /* 2 * Copyright 2018 Google Inc. 3 * 4 * Use of this source code is governed by a BSD-style license that can be 5 * found in the LICENSE file. 6 */ 7 8 #ifndef SKSL_JIT 9 #define SKSL_JIT 10 11 #ifdef SK_LLVM_AVAILABLE 12 13 #include "src/sksl/ir/SkSLBinaryExpression.h" 14 #include "src/sksl/ir/SkSLBreakStatement.h" 15 #include "src/sksl/ir/SkSLContinueStatement.h" 16 #include "src/sksl/ir/SkSLDoStatement.h" 17 #include "src/sksl/ir/SkSLExpression.h" 18 #include "src/sksl/ir/SkSLForStatement.h" 19 #include "src/sksl/ir/SkSLFunctionCall.h" 20 #include "src/sksl/ir/SkSLFunctionDefinition.h" 21 #include "src/sksl/ir/SkSLIfStatement.h" 22 #include "src/sksl/ir/SkSLIndexExpression.h" 23 #include "src/sksl/ir/SkSLPostfixExpression.h" 24 #include "src/sksl/ir/SkSLPrefixExpression.h" 25 #include "src/sksl/ir/SkSLProgram.h" 26 #include "src/sksl/ir/SkSLReturnStatement.h" 27 #include "src/sksl/ir/SkSLStatement.h" 28 #include "src/sksl/ir/SkSLSwizzle.h" 29 #include "src/sksl/ir/SkSLTernaryExpression.h" 30 #include "src/sksl/ir/SkSLVarDeclarationsStatement.h" 31 #include "src/sksl/ir/SkSLVariableReference.h" 32 #include "src/sksl/ir/SkSLWhileStatement.h" 33 34 #include "llvm-c/Analysis.h" 35 #include "llvm-c/Core.h" 36 #include "llvm-c/OrcBindings.h" 37 #include "llvm-c/Support.h" 38 #include "llvm-c/Target.h" 39 #include "llvm-c/Transforms/PassManagerBuilder.h" 40 #include "llvm-c/Types.h" 41 #include <stack> 42 43 class SkRasterPipeline; 44 45 namespace SkSL { 46 47 struct AppendStage; 48 49 /** 50 * A just-in-time compiler for SkSL code which uses an LLVM backend. Only available when the 51 * skia_llvm_path gn arg is set. 52 * 53 * Example of using SkSLJIT to set up an SkJumper pipeline stage: 54 * 55 * #ifdef SK_LLVM_AVAILABLE 56 * SkSL::Compiler compiler; 57 * SkSL::Program::Settings settings; 58 * std::unique_ptr<SkSL::Program> program = compiler.convertProgram( 59 SkSL::Program::kPipelineStage_Kind, 60 * "void swap(int x, int y, inout float4 color) {" 61 * " color.rb = color.br;" 62 * "}", 63 * settings); 64 * if (!program) { 65 * printf("%s\n", compiler.errorText().c_str()); 66 * abort(); 67 * } 68 * SkSL::JIT& jit = *scratch->make<SkSL::JIT>(&compiler); 69 * std::unique_ptr<SkSL::JIT::Module> module = jit.compile(std::move(program)); 70 * void* func = module->getJumperStage("swap"); 71 * p->append(func, nullptr); 72 * #endif 73 */ 74 class JIT { 75 typedef int StackIndex; 76 77 public: 78 class Module { 79 public: 80 /** 81 * Returns the address of a symbol in the module. 82 */ 83 void* getSymbol(const char* name); 84 85 /** 86 * Returns the address of a function as an SkJumper pipeline stage. The function must have 87 * the signature void <name>(int x, int y, inout float4 color). The returned function will 88 * have the correct signature to function as an SkJumper stage (meaning it will actually 89 * have a different signature at runtime, accepting vector parameters and operating on 90 * multiple pixels simultaneously as is normal for SkJumper stages). 91 */ 92 void* getJumperStage(const char* name); 93 ~Module()94 ~Module() { 95 LLVMOrcDisposeSharedModuleRef(fSharedModule); 96 } 97 98 private: Module(std::unique_ptr<Program> program,LLVMSharedModuleRef sharedModule,LLVMOrcJITStackRef jitStack)99 Module(std::unique_ptr<Program> program, 100 LLVMSharedModuleRef sharedModule, 101 LLVMOrcJITStackRef jitStack) 102 : fProgram(std::move(program)) 103 , fSharedModule(sharedModule) 104 , fJITStack(jitStack) {} 105 106 std::unique_ptr<Program> fProgram; 107 LLVMSharedModuleRef fSharedModule; 108 LLVMOrcJITStackRef fJITStack; 109 110 friend class JIT; 111 }; 112 113 JIT(Compiler* compiler); 114 115 ~JIT(); 116 117 /** 118 * Just-in-time compiles an SkSL program and returns the resulting Module. The JIT must not be 119 * destroyed before all of its Modules are destroyed. 120 */ 121 std::unique_ptr<Module> compile(std::unique_ptr<Program> program); 122 123 private: 124 static constexpr int CHANNELS = 4; 125 126 enum TypeKind { 127 kFloat_TypeKind, 128 kInt_TypeKind, 129 kUInt_TypeKind, 130 kBool_TypeKind 131 }; 132 133 class LValue { 134 public: ~LValue()135 virtual ~LValue() {} 136 137 virtual LLVMValueRef load(LLVMBuilderRef builder) = 0; 138 139 virtual void store(LLVMBuilderRef builder, LLVMValueRef value) = 0; 140 }; 141 142 void addBuiltinFunction(const char* ourName, const char* realName, LLVMTypeRef returnType, 143 std::vector<LLVMTypeRef> parameters); 144 145 void loadBuiltinFunctions(); 146 147 void setBlock(LLVMBuilderRef builder, LLVMBasicBlockRef block); 148 149 LLVMTypeRef getType(const Type& type); 150 151 TypeKind typeKind(const Type& type); 152 153 std::unique_ptr<LValue> getLValue(LLVMBuilderRef builder, const Expression& expr); 154 155 void vectorize(LLVMBuilderRef builder, LLVMValueRef* value, int columns); 156 157 void vectorize(LLVMBuilderRef builder, const BinaryExpression& b, LLVMValueRef* left, 158 LLVMValueRef* right); 159 160 LLVMValueRef compileBinary(LLVMBuilderRef builder, const BinaryExpression& b); 161 162 LLVMValueRef compileConstructor(LLVMBuilderRef builder, const Constructor& c); 163 164 LLVMValueRef compileFunctionCall(LLVMBuilderRef builder, const FunctionCall& fc); 165 166 LLVMValueRef compileIndex(LLVMBuilderRef builder, const IndexExpression& v); 167 168 LLVMValueRef compilePostfix(LLVMBuilderRef builder, const PostfixExpression& p); 169 170 LLVMValueRef compilePrefix(LLVMBuilderRef builder, const PrefixExpression& p); 171 172 LLVMValueRef compileSwizzle(LLVMBuilderRef builder, const Swizzle& s); 173 174 LLVMValueRef compileVariableReference(LLVMBuilderRef builder, const VariableReference& v); 175 176 LLVMValueRef compileTernary(LLVMBuilderRef builder, const TernaryExpression& t); 177 178 LLVMValueRef compileExpression(LLVMBuilderRef builder, const Expression& expr); 179 180 void appendStage(LLVMBuilderRef builder, const AppendStage& a); 181 182 void compileBlock(LLVMBuilderRef builder, const Block& block); 183 184 void compileBreak(LLVMBuilderRef builder, const BreakStatement& b); 185 186 void compileContinue(LLVMBuilderRef builder, const ContinueStatement& c); 187 188 void compileDo(LLVMBuilderRef builder, const DoStatement& d); 189 190 void compileFor(LLVMBuilderRef builder, const ForStatement& f); 191 192 void compileIf(LLVMBuilderRef builder, const IfStatement& i); 193 194 void compileReturn(LLVMBuilderRef builder, const ReturnStatement& r); 195 196 void compileVarDeclarations(LLVMBuilderRef builder, const VarDeclarationsStatement& decls); 197 198 void compileWhile(LLVMBuilderRef builder, const WhileStatement& w); 199 200 void compileStatement(LLVMBuilderRef builder, const Statement& stmt); 201 202 // The "Vector" variants of functions attempt to compile a given expression or statement as part 203 // of a vectorized SkJumper stage function - that is, with r, g, b, and a each being vectors of 204 // fVectorCount floats. So a statement like "color.r = 0;" looks like it modifies a single 205 // channel of a single pixel, but the compiled code will actually modify the red channel of 206 // fVectorCount pixels at once. 207 // 208 // As not everything can be vectorized, these calls return a bool to indicate whether they were 209 // successful. If anything anywhere in the function cannot be vectorized, the JIT will fall back 210 // to looping over the pixels instead. 211 // 212 // Since we process multiple pixels at once, and each pixel consists of multiple color channels, 213 // expressions may effectively result in a vector-of-vectors. We produce zero to four outputs 214 // when compiling expression, each of which is a vector, so that e.g. float2(1, 0) actually 215 // produces two vectors, one containing all 1s, the other all 0s. The out parameter always 216 // allows for 4 channels, but the functions produce 0 to 4 channels depending on the type they 217 // are operating on. Thus evaluating "color.rgb" actually fills in out[0] through out[2], 218 // leaving out[3] uninitialized. 219 // As the number of outputs can be inferred from the type of the expression, it is not 220 // explicitly signalled anywhere. 221 bool compileVectorBinary(LLVMBuilderRef builder, const BinaryExpression& b, 222 LLVMValueRef out[CHANNELS]); 223 224 bool compileVectorConstructor(LLVMBuilderRef builder, const Constructor& c, 225 LLVMValueRef out[CHANNELS]); 226 227 bool compileVectorFloatLiteral(LLVMBuilderRef builder, const FloatLiteral& f, 228 LLVMValueRef out[CHANNELS]); 229 230 bool compileVectorSwizzle(LLVMBuilderRef builder, const Swizzle& s, 231 LLVMValueRef out[CHANNELS]); 232 233 bool compileVectorVariableReference(LLVMBuilderRef builder, const VariableReference& v, 234 LLVMValueRef out[CHANNELS]); 235 236 bool compileVectorExpression(LLVMBuilderRef builder, const Expression& expr, 237 LLVMValueRef out[CHANNELS]); 238 239 bool getVectorLValue(LLVMBuilderRef builder, const Expression& e, LLVMValueRef out[CHANNELS]); 240 241 /** 242 * Evaluates the left and right operands of a binary operation, promoting one of them to a 243 * vector if necessary to make the types match. 244 */ 245 bool getVectorBinaryOperands(LLVMBuilderRef builder, const Expression& left, 246 LLVMValueRef outLeft[CHANNELS], const Expression& right, 247 LLVMValueRef outRight[CHANNELS]); 248 249 bool compileVectorStatement(LLVMBuilderRef builder, const Statement& stmt); 250 251 /** 252 * Returns true if this function has the signature void(int, int, inout float4) and thus can be 253 * used as an SkJumper stage. 254 */ 255 bool hasStageSignature(const FunctionDeclaration& f); 256 257 /** 258 * Attempts to compile a vectorized stage function, returning true on success. A stage function 259 * of e.g. "color.r = 0;" will produce code which sets the entire red vector to zeros in a 260 * single instruction, thus calculating several pixels at once. 261 */ 262 bool compileStageFunctionVector(const FunctionDefinition& f, LLVMValueRef newFunc); 263 264 /** 265 * Fallback function which loops over the pixels, for when vectorization fails. A stage function 266 * of e.g. "color.r = 0;" will produce a loop which iterates over the entries in the red vector, 267 * setting each one to zero individually. 268 */ 269 void compileStageFunctionLoop(const FunctionDefinition& f, LLVMValueRef newFunc); 270 271 /** 272 * Called when compiling a function which has the signature of an SkJumper stage. Produces a 273 * version of the function which can be plugged into SkJumper (thus having a signature which 274 * accepts four vectors, one for each color channel, containing the color data of multiple 275 * pixels at once). To go from SkSL code which operates on a single pixel at a time to CPU code 276 * which operates on multiple pixels at once, the code is either vectorized using 277 * compileStageFunctionVector or wrapped in a loop using compileStageFunctionLoop. 278 */ 279 LLVMValueRef compileStageFunction(const FunctionDefinition& f); 280 281 /** 282 * Compiles an SkSL function to an LLVM function. If the function has the signature of an 283 * SkJumper stage, it will *also* be compiled by compileStageFunction, resulting in both a stage 284 * and non-stage version of the function. 285 */ 286 LLVMValueRef compileFunction(const FunctionDefinition& f); 287 288 void createModule(); 289 290 void optimize(); 291 292 bool isColorRef(const Expression& expr); 293 294 static uint64_t resolveSymbol(const char* name, JIT* jit); 295 296 const char* fCPU; 297 int fVectorCount; 298 Compiler& fCompiler; 299 std::unique_ptr<Program> fProgram; 300 LLVMContextRef fContext; 301 LLVMModuleRef fModule; 302 LLVMSharedModuleRef fSharedModule; 303 LLVMOrcJITStackRef fJITStack; 304 LLVMValueRef fCurrentFunction; 305 LLVMBasicBlockRef fAllocaBlock; 306 LLVMBasicBlockRef fCurrentBlock; 307 LLVMTypeRef fVoidType; 308 LLVMTypeRef fInt1Type; 309 LLVMTypeRef fInt1VectorType; 310 LLVMTypeRef fInt1Vector2Type; 311 LLVMTypeRef fInt1Vector3Type; 312 LLVMTypeRef fInt1Vector4Type; 313 LLVMTypeRef fInt8Type; 314 LLVMTypeRef fInt8PtrType; 315 LLVMTypeRef fInt32Type; 316 LLVMTypeRef fInt32VectorType; 317 LLVMTypeRef fInt32Vector2Type; 318 LLVMTypeRef fInt32Vector3Type; 319 LLVMTypeRef fInt32Vector4Type; 320 LLVMTypeRef fInt64Type; 321 LLVMTypeRef fSizeTType; 322 LLVMTypeRef fFloat32Type; 323 LLVMTypeRef fFloat32VectorType; 324 LLVMTypeRef fFloat32Vector2Type; 325 LLVMTypeRef fFloat32Vector3Type; 326 LLVMTypeRef fFloat32Vector4Type; 327 // Our SkSL stage functions have a single float4 for color, but the actual SkJumper stage 328 // function has four separate vectors, one for each channel. These four values are references to 329 // the red, green, blue, and alpha vectors respectively. 330 LLVMValueRef fChannels[CHANNELS]; 331 // when processing a stage function, this points to the SkSL color parameter (an inout float4) 332 const Variable* fColorParam; 333 std::unordered_map<const FunctionDeclaration*, LLVMValueRef> fFunctions; 334 std::unordered_map<const Variable*, LLVMValueRef> fVariables; 335 // LLVM function parameters are read-only, so when modifying function parameters we need to 336 // first promote them to variables. This keeps track of which parameters have been promoted. 337 std::set<const Variable*> fPromotedParameters; 338 std::vector<LLVMBasicBlockRef> fBreakTarget; 339 std::vector<LLVMBasicBlockRef> fContinueTarget; 340 341 LLVMValueRef fFoldAnd2Func; 342 LLVMValueRef fFoldOr2Func; 343 LLVMValueRef fFoldAnd3Func; 344 LLVMValueRef fFoldOr3Func; 345 LLVMValueRef fFoldAnd4Func; 346 LLVMValueRef fFoldOr4Func; 347 LLVMValueRef fAppendFunc; 348 LLVMValueRef fAppendCallbackFunc; 349 LLVMValueRef fDebugFunc; 350 }; 351 352 } // namespace 353 354 #endif // SK_LLVM_AVAILABLE 355 356 #endif // SKSL_JIT 357