1cb93a386Sopenharmony_ci/* 2cb93a386Sopenharmony_ci * Copyright 2021 Google LLC 3cb93a386Sopenharmony_ci * 4cb93a386Sopenharmony_ci * Use of this source code is governed by a BSD-style license that can be 5cb93a386Sopenharmony_ci * found in the LICENSE file. 6cb93a386Sopenharmony_ci */ 7cb93a386Sopenharmony_ci 8cb93a386Sopenharmony_ci#include "src/sksl/ir/SkSLConstructorMatrixResize.h" 9cb93a386Sopenharmony_ci 10cb93a386Sopenharmony_ci#include "src/sksl/ir/SkSLConstructor.h" 11cb93a386Sopenharmony_ci#include "src/sksl/ir/SkSLType.h" 12cb93a386Sopenharmony_ci 13cb93a386Sopenharmony_cinamespace SkSL { 14cb93a386Sopenharmony_ci 15cb93a386Sopenharmony_cistd::unique_ptr<Expression> ConstructorMatrixResize::Make(const Context& context, 16cb93a386Sopenharmony_ci int line, 17cb93a386Sopenharmony_ci const Type& type, 18cb93a386Sopenharmony_ci std::unique_ptr<Expression> arg) { 19cb93a386Sopenharmony_ci SkASSERT(type.isMatrix()); 20cb93a386Sopenharmony_ci SkASSERT(type.isAllowedInES2(context)); 21cb93a386Sopenharmony_ci SkASSERT(arg->type().componentType() == type.componentType()); 22cb93a386Sopenharmony_ci 23cb93a386Sopenharmony_ci // If the matrix isn't actually changing size, return it as-is. 24cb93a386Sopenharmony_ci if (type.rows() == arg->type().rows() && type.columns() == arg->type().columns()) { 25cb93a386Sopenharmony_ci return arg; 26cb93a386Sopenharmony_ci } 27cb93a386Sopenharmony_ci 28cb93a386Sopenharmony_ci return std::make_unique<ConstructorMatrixResize>(line, type, std::move(arg)); 29cb93a386Sopenharmony_ci} 30cb93a386Sopenharmony_ci 31cb93a386Sopenharmony_ciskstd::optional<double> ConstructorMatrixResize::getConstantValue(int n) const { 32cb93a386Sopenharmony_ci int rows = this->type().rows(); 33cb93a386Sopenharmony_ci int row = n % rows; 34cb93a386Sopenharmony_ci int col = n / rows; 35cb93a386Sopenharmony_ci 36cb93a386Sopenharmony_ci SkASSERT(col >= 0); 37cb93a386Sopenharmony_ci SkASSERT(row >= 0); 38cb93a386Sopenharmony_ci SkASSERT(col < this->type().columns()); 39cb93a386Sopenharmony_ci SkASSERT(row < this->type().rows()); 40cb93a386Sopenharmony_ci 41cb93a386Sopenharmony_ci // GLSL resize matrices are of the form: 42cb93a386Sopenharmony_ci // |m m 0| 43cb93a386Sopenharmony_ci // |m m 0| 44cb93a386Sopenharmony_ci // |0 0 1| 45cb93a386Sopenharmony_ci // Where `m` is the matrix being wrapped, and other cells contain the identity matrix. 46cb93a386Sopenharmony_ci 47cb93a386Sopenharmony_ci // Forward `getConstantValue` to the wrapped matrix if the position is in its bounds. 48cb93a386Sopenharmony_ci if (col < this->argument()->type().columns() && row < this->argument()->type().rows()) { 49cb93a386Sopenharmony_ci // Recalculate `n` in terms of the inner matrix's dimensions. 50cb93a386Sopenharmony_ci n = row + (col * this->argument()->type().rows()); 51cb93a386Sopenharmony_ci return this->argument()->getConstantValue(n); 52cb93a386Sopenharmony_ci } 53cb93a386Sopenharmony_ci 54cb93a386Sopenharmony_ci // Synthesize an identity matrix for out-of-bounds positions. 55cb93a386Sopenharmony_ci return (col == row) ? 1.0 : 0.0; 56cb93a386Sopenharmony_ci} 57cb93a386Sopenharmony_ci 58cb93a386Sopenharmony_ci} // namespace SkSL 59