/**
|
* @license
|
* Copyright 2021 Google LLC. 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.
|
* =============================================================================
|
*/
|
export class TransformProgram {
|
constructor(imageHeight, imageWidth, interpolation, fillMode, fillValue, outShape) {
|
this.variableNames = ['Image', 'Transforms'];
|
this.outputShape = outShape;
|
const interpolationModeId = interpolation === 'nearest' ? 1 : 2;
|
let fillModeId;
|
switch (fillMode) {
|
case 'constant':
|
fillModeId = 1;
|
break;
|
case 'reflect':
|
fillModeId = 2;
|
break;
|
case 'wrap':
|
fillModeId = 3;
|
break;
|
case 'nearest':
|
fillModeId = 4;
|
break;
|
default:
|
fillModeId = 1;
|
break;
|
}
|
this.userCode = `
|
float mapCoord(float outCoord, float len) {
|
float inCoord = outCoord;
|
if(${fillModeId} == 2) {
|
if (inCoord < 0.0) {
|
if (len <= 1.0) {
|
inCoord = 0.0;
|
} else {
|
float sz2 = 2.0 * len;
|
if (inCoord < sz2) {
|
inCoord = sz2 * float(int(float(-inCoord / sz2))) +
|
inCoord;
|
}
|
inCoord = inCoord < -len ? inCoord + sz2 : -inCoord - 1.0;
|
}
|
} else if (inCoord > len - 1.0) {
|
if (len <= 1.0) {
|
inCoord = 0.0;
|
} else {
|
float sz2 = 2.0 * len;
|
inCoord -= sz2 * float(int(float(inCoord / sz2)));
|
if (inCoord >= len) {
|
inCoord = sz2 - inCoord - 1.0;
|
}
|
}
|
}
|
return clamp(inCoord, 0.0, len - 1.0);
|
} else if (${fillModeId} == 3) {
|
if (inCoord < 0.0) {
|
if (len <= 1.0) {
|
inCoord = 0.0;
|
} else {
|
float sz = len - 1.0;
|
inCoord += len * (float(int(float(-inCoord / sz))) + 1.0);
|
}
|
} else if (inCoord > len - 1.0) {
|
if (len <= 1.0) {
|
inCoord = 0.0;
|
} else {
|
float sz = len - 1.0;
|
inCoord -= len * float(int(float(inCoord / sz)));
|
}
|
}
|
return clamp(inCoord, 0.0, len - 1.0);
|
} else if (${fillModeId} == 4) {
|
return clamp(outCoord, 0.0, len - 1.0);
|
} else {
|
return outCoord;
|
}
|
}
|
|
float readWithFillValue(int batch, int coordY, int coordX,
|
int channel) {
|
float outputValue;
|
if (0 <= coordY && coordY < ${imageHeight} && 0 <= coordX && coordX < ${imageWidth}) {
|
outputValue = getImage(batch, coordY, coordX, channel);
|
} else {
|
outputValue = float(${fillValue});
|
}
|
return outputValue;
|
}
|
|
void main() {
|
ivec4 coords = getOutputCoords();
|
float outputValue;
|
int batch = coords[0];
|
int x = coords[2];
|
int y = coords[1];
|
int channel = coords[3];
|
float xf = float(x);
|
float yf = float(y);
|
float a1 = getTransforms(batch, 0);
|
float a2 = getTransforms(batch, 1);
|
float a3 = getTransforms(batch, 2);
|
float b1 = getTransforms(batch, 3);
|
float b2 = getTransforms(batch, 4);
|
float b3 = getTransforms(batch, 5);
|
float c1 = getTransforms(batch, 6);
|
float c2 = getTransforms(batch, 7);
|
float projection = c1 * xf + c2 * yf + 1.0;
|
if (projection == 0.0) {
|
outputValue = float(${fillValue});
|
} else {
|
float inX = (a1 * xf + a2 * yf + a3) / projection;
|
float inY = (b1 * xf + b2 * yf + b3) / projection;
|
float mapX = mapCoord(inX, float(${imageWidth}));
|
float mapY = mapCoord(inY, float(${imageHeight}));
|
|
if (${interpolationModeId} == 1) {
|
int coordY = int(round(mapY));
|
int coordX = int(round(mapX));
|
outputValue = readWithFillValue(batch, coordY, coordX,
|
channel);
|
} else {
|
float yFloor = floor(mapY);
|
float xFloor = floor(mapX);
|
float yCeil = yFloor + 1.0;
|
float xCeil = xFloor + 1.0;
|
float valueYFloor = (xCeil - mapX) *
|
readWithFillValue(batch, int(yFloor), int(xFloor), channel) +
|
(mapX - xFloor) *
|
readWithFillValue(batch, int(yFloor), int(xCeil), channel);
|
float valueYCeil = (xCeil - mapX) *
|
readWithFillValue(batch, int(yCeil), int(xFloor), channel) +
|
(mapX - xFloor) *
|
readWithFillValue(batch, int(yCeil), int(xCeil), channel);
|
outputValue = (yCeil - mapY) * valueYFloor +
|
(mapY - yFloor) * valueYCeil;
|
}
|
}
|
setOutput(outputValue);
|
}
|
`;
|
}
|
}
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transform_gpu.js","sourceRoot":"","sources":["../../../../../tfjs-backend-webgl/src/transform_gpu.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAIH,MAAM,OAAO,gBAAgB;IAK3B,YACI,WAAmB,EAAE,UAAkB,EACvC,aAAmC,EACnC,QAA+C,EAAE,SAAiB,EAClE,QAA0C;QAR9C,kBAAa,GAAG,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QAStC,IAAI,CAAC,WAAW,GAAG,QAAQ,CAAC;QAC5B,MAAM,mBAAmB,GAAG,aAAa,KAAK,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAChE,IAAI,UAAU,CAAC;QACf,QAAQ,QAAQ,EAAE;YAChB,KAAK,UAAU;gBACb,UAAU,GAAG,CAAC,CAAC;gBACf,MAAM;YACR,KAAK,SAAS;gBACZ,UAAU,GAAG,CAAC,CAAC;gBACf,MAAM;YACR,KAAK,MAAM;gBACT,UAAU,GAAG,CAAC,CAAC;gBACf,MAAM;YACR,KAAK,SAAS;gBACZ,UAAU,GAAG,CAAC,CAAC;gBACf,MAAM;YACR;gBACE,UAAU,GAAG,CAAC,CAAC;gBACf,MAAM;SACT;QACD,IAAI,CAAC,QAAQ,GAAG;;;mBAGD,UAAU;;;;;;;;;;;;;;;;;;;;;;;;2BAwBF,UAAU;;;;;;;;;;;;;;;;;2BAiBV,UAAU;;;;;;;;;;4CAW7B,WAAW,+BAA+B,UAAU;;;sCAGtB,SAAS;;;;;;;;;;;;;;;;;;;;;;;;sCAwBT,SAAS;;;;mDAII,UAAU;mDACV,WAAW;;sBAExC,mBAAmB;;;;;;;;;;;;;;;;;;;;;;;;SAwBhC,CAAC;IACR,CAAC;CACF","sourcesContent":["/**\n * @license\n * Copyright 2021 Google LLC. All Rights Reserved.\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n * =============================================================================\n */\n\nimport {GPGPUProgram} from './gpgpu_math';\n\nexport class TransformProgram implements GPGPUProgram {\n  variableNames = ['Image', 'Transforms'];\n  outputShape: number[];\n  userCode: string;\n\n  constructor(\n      imageHeight: number, imageWidth: number,\n      interpolation: 'nearest'|'bilinear',\n      fillMode: 'constant'|'reflect'|'wrap'|'nearest', fillValue: number,\n      outShape: [number, number, number, number]) {\n    this.outputShape = outShape;\n    const interpolationModeId = interpolation === 'nearest' ? 1 : 2;\n    let fillModeId;\n    switch (fillMode) {\n      case 'constant':\n        fillModeId = 1;\n        break;\n      case 'reflect':\n        fillModeId = 2;\n        break;\n      case 'wrap':\n        fillModeId = 3;\n        break;\n      case 'nearest':\n        fillModeId = 4;\n        break;\n      default:\n        fillModeId = 1;\n        break;\n    }\n    this.userCode = `\n            float mapCoord(float outCoord, float len) {\n              float inCoord = outCoord;\n              if(${fillModeId} == 2) {\n                if (inCoord < 0.0) {\n                  if (len <= 1.0) {\n                    inCoord = 0.0;\n                  } else {\n                    float sz2 = 2.0 * len;\n                    if (inCoord < sz2) {\n                      inCoord = sz2 * float(int(float(-inCoord / sz2))) +\n                      inCoord;\n                    }\n                    inCoord = inCoord < -len ? inCoord + sz2 : -inCoord - 1.0;\n                  }\n                } else if (inCoord > len - 1.0) {\n                  if (len <= 1.0) {\n                    inCoord = 0.0;\n                  } else {\n                    float sz2 = 2.0 * len;\n                    inCoord -= sz2 * float(int(float(inCoord / sz2)));\n                    if (inCoord >= len) {\n                      inCoord = sz2 - inCoord - 1.0;\n                    }\n                  }\n                }\n                return clamp(inCoord, 0.0, len - 1.0);\n              } else if (${fillModeId} == 3) {\n                if (inCoord < 0.0) {\n                  if (len <= 1.0) {\n                    inCoord = 0.0;\n                  } else {\n                    float sz = len - 1.0;\n                    inCoord += len * (float(int(float(-inCoord / sz))) + 1.0);\n                  }\n                } else if (inCoord > len - 1.0) {\n                  if (len <= 1.0) {\n                    inCoord = 0.0;\n                  } else {\n                    float sz = len - 1.0;\n                    inCoord -= len * float(int(float(inCoord / sz)));\n                  }\n                }\n                return clamp(inCoord, 0.0, len - 1.0);\n              } else if (${fillModeId} == 4) {\n                return clamp(outCoord, 0.0, len - 1.0);\n              } else {\n                return outCoord;\n              }\n            }\n\n            float readWithFillValue(int batch, int coordY, int coordX,\n              int channel) {\n              float outputValue;\n              if (0 <= coordY && coordY < ${\n        imageHeight} && 0 <= coordX && coordX < ${imageWidth}) {\n                  outputValue = getImage(batch, coordY, coordX, channel);\n              } else {\n                outputValue = float(${fillValue});\n              }\n              return outputValue;\n            }\n\n            void main() {\n              ivec4 coords = getOutputCoords();\n              float outputValue;\n              int batch = coords[0];\n              int x = coords[2];\n              int y = coords[1];\n              int channel = coords[3];\n              float xf = float(x);\n              float yf = float(y);\n              float a1 = getTransforms(batch, 0);\n              float a2 = getTransforms(batch, 1);\n              float a3 = getTransforms(batch, 2);\n              float b1 = getTransforms(batch, 3);\n              float b2 = getTransforms(batch, 4);\n              float b3 = getTransforms(batch, 5);\n              float c1 = getTransforms(batch, 6);\n              float c2 = getTransforms(batch, 7);\n              float projection = c1 * xf + c2 * yf + 1.0;\n              if (projection == 0.0) {\n                outputValue = float(${fillValue});\n              } else {\n                float inX = (a1 * xf + a2 * yf + a3) / projection;\n                float inY = (b1 * xf + b2 * yf + b3) / projection;\n                float mapX = mapCoord(inX, float(${imageWidth}));\n                float mapY = mapCoord(inY, float(${imageHeight}));\n\n                if (${interpolationModeId} == 1) {\n                  int coordY = int(round(mapY));\n                  int coordX = int(round(mapX));\n                  outputValue = readWithFillValue(batch, coordY, coordX,\n                    channel);\n                } else {\n                  float yFloor = floor(mapY);\n                  float xFloor = floor(mapX);\n                  float yCeil = yFloor + 1.0;\n                  float xCeil = xFloor + 1.0;\n                  float valueYFloor = (xCeil - mapX) *\n                  readWithFillValue(batch, int(yFloor), int(xFloor), channel) +\n                  (mapX - xFloor) *\n                  readWithFillValue(batch, int(yFloor), int(xCeil), channel);\n                  float valueYCeil = (xCeil - mapX) *\n                  readWithFillValue(batch, int(yCeil), int(xFloor), channel) +\n                  (mapX - xFloor) *\n                  readWithFillValue(batch, int(yCeil), int(xCeil), channel);\n                  outputValue = (yCeil - mapY) * valueYFloor +\n                  (mapY - yFloor) * valueYCeil;\n                }\n              }\n              setOutput(outputValue);\n            }\n        `;\n  }\n}\n"]}
|