UNPKG

@tensorflow/tfjs-core

Version:

Hardware-accelerated JavaScript library for machine intelligence

73 lines (66 loc) 2.4 kB
/** * @license * Copyright 2019 Google Inc. All Rights Reserved. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ============================================================================= */ import {getVecChannels} from '../packing_util'; import {GPGPUProgram} from './gpgpu_math'; import {getCoordsDataType} from './shader_compiler'; export class TransposePackedProgram implements GPGPUProgram { variableNames = ['A']; outputShape: number[]; userCode: string; rank: number; usesPackedTextures = true; constructor(aShape: number[], newDim: number[]) { const outputShape: number[] = new Array(aShape.length); for (let i = 0; i < outputShape.length; i++) { outputShape[i] = aShape[newDim[i]]; } this.outputShape = outputShape; this.rank = outputShape.length; if (this.rank > 6) { throw Error( `Packed transpose for rank ${this.rank} is not yet supported.`); } const dtype = getCoordsDataType(this.rank); const outputOrder = getVecChannels('rc', this.rank); const switchedOrder = new Array(this.rank); for (let i = 0; i < newDim.length; i++) { switchedOrder[newDim[i]] = outputOrder[i]; } const innerDims = `vec2(${switchedOrder.slice(-2).join()})`; const nextColumn = `++${outputOrder[this.rank - 1]} < ${outputShape[this.rank - 1]}`; const getc = `getChannel(getA(${switchedOrder.join()}), ${innerDims})`; this.userCode = ` void main() { ${dtype} rc = getOutputCoords(); vec4 result = vec4(0.); result[0] = ${getc}; if(${nextColumn}) { result[1] = ${getc}; } --${outputOrder[this.rank - 1]}; if(++${outputOrder[this.rank - 2]} < ${outputShape[this.rank - 2]}) { result[2] = ${getc}; if(${nextColumn}) { result[3] = ${getc}; } } setOutput(result); } `; } }