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