/**
|
* @license
|
* Copyright 2018 Google LLC
|
*
|
* Use of this source code is governed by an MIT-style
|
* license that can be found in the LICENSE file or at
|
* https://opensource.org/licenses/MIT.
|
* =============================================================================
|
*/
|
/**
|
* Normalization layers.
|
*/
|
import * as tfc from '@tensorflow/tfjs-core';
|
import { moments, reshape, serialization, tidy, util } from '@tensorflow/tfjs-core';
|
import { getConstraint, serializeConstraint } from '../constraints';
|
import { InputSpec, Layer } from '../engine/topology';
|
import { NotImplementedError, ValueError } from '../errors';
|
import { getInitializer, serializeInitializer } from '../initializers';
|
import { getRegularizer, serializeRegularizer } from '../regularizers';
|
import * as generic_utils from '../utils/generic_utils';
|
import * as math_utils from '../utils/math_utils';
|
import { getExactlyOneShape, getExactlyOneTensor } from '../utils/types_utils';
|
/**
|
* Applies batch normalization on x given mean, var, beta and gamma.
|
*
|
* I.e. returns:
|
* `output = (x - mean) / (sqrt(var) + epsilon) * gamma + beta`
|
*
|
* @param x Input tensor.
|
* @param mean Mean of batch.
|
* @param variance Variance of batch.
|
* @param beta Tensor with which to center the input.
|
* @param gamma Tensor by which to scale the input.
|
* @param epsilon Fuzz factor.
|
* @returns The result of the batch normalization.
|
*/
|
export function batchNormalization(x, mean, variance, beta, gamma, epsilon = 1e-3) {
|
let out;
|
if (x.rank === 2) {
|
out = tfc.batchNorm2d(x, mean, variance, beta, gamma, epsilon);
|
}
|
else if (x.rank === 3) {
|
// TODO(cais): Check rank; give proper error message.
|
out = tfc.batchNorm3d(x, mean, variance, beta, gamma, epsilon);
|
}
|
else if (x.rank === 4) {
|
out = tfc.batchNorm4d(x, mean, variance, beta, gamma, epsilon);
|
}
|
else {
|
throw new NotImplementedError(`batchNormalization is not implemented for array of rank ${x.rank} ` +
|
`yet`);
|
}
|
return out;
|
}
|
/**
|
* Non-broadcasting batch normalization for use in training (not inference).
|
*
|
* The input is normalized to zero mean and unit variance along the
|
* `reductionAxes`, followed by scaling with `gamma` and shifted by `beta`.
|
* The result of that is returned as the first element
|
* of the returned `Array`. The other two elements are the mean and variance,
|
* respectively.
|
*
|
* @param x Input tensor to be normalized.
|
* @param gamma Tensor by which to scale the input.
|
* @param beta Tensor by which to center the input.
|
* @param reductionAxes Axes over which to normalize.
|
* @param epsilon Fuzz factor.
|
* @returns An `Array` of three `Tensors`:
|
* [normalized tensor, mean of input, variance of input].
|
*/
|
function regularNormalizeBatchInTraining(x, gamma, beta, reductionAxes, epsilon = 1e-3) {
|
return tidy(() => {
|
const meanAndVariance = tfc.moments(x, reductionAxes);
|
const mean = meanAndVariance.mean;
|
const variance = meanAndVariance.variance;
|
const normed = batchNormalization(x, mean, variance, beta, gamma, epsilon);
|
return [normed, mean, variance];
|
});
|
}
|
/**
|
* Broadcasting batch normalization for use in training (not inference).
|
*
|
* The input is normalized to zero mean and unit variance along the
|
* `reductionAxes`, followed by scaling with `gamma` and shifted by `beta`.
|
* The result of that is returned as the first element
|
* of the returned `Array`. The other two elements are the mean and variance,
|
* respectively.
|
*
|
* @param x Input tensor to be normalized.
|
* @param gamma Tensor by which to scale the input.
|
* @param beta Tensor by which to center the input.
|
* @param reductionAxes Axes over which to normalize.
|
* @param epsilon Fuzz factor.
|
* @returns An `Array` of three `Tensors`:
|
* [normalized tensor, mean of input, variance of input].
|
*/
|
function broadcastNormalizeBatchInTraining(x, gamma, beta, reductionAxes, epsilon = 1e-3) {
|
return tidy(() => {
|
const meanAndVariance = tfc.moments(x, reductionAxes);
|
const mean = meanAndVariance.mean;
|
const variance = meanAndVariance.variance;
|
const targetShape = [];
|
for (const axis of math_utils.range(0, x.rank)) {
|
if (reductionAxes.indexOf(axis) !== -1) {
|
targetShape.push(1);
|
}
|
else {
|
targetShape.push(x.shape[axis]);
|
}
|
}
|
const broadcastMean = reshape(mean, targetShape);
|
const broadcastVariance = reshape(variance, targetShape);
|
const broadcastGamma = gamma == null ? null : reshape(gamma, targetShape);
|
const broadcastBeta = beta == null ? null : reshape(beta, targetShape);
|
const normed = batchNormalization(x, broadcastMean, broadcastVariance, broadcastBeta, broadcastGamma, epsilon);
|
return [normed, mean, variance];
|
});
|
}
|
/**
|
* Batch normalization for use in training (not inference).
|
*
|
* @param x Input tensor to be normalized.
|
* @param gamma Tensor by which to scale the input.
|
* @param beta Tensor by which to center the input.
|
* @param reductionAxes Axes over which to normalize.
|
* @param epsilon Fuzz factor.
|
* @returns An `Array` of three `Tensors`:
|
* [normalized tensor, mean of input, variance of input].
|
*/
|
export function normalizeBatchInTraining(x, gamma, beta, reductionAxes, epsilon = 1e-3) {
|
if (util.arraysEqual(reductionAxes.slice().sort(), math_utils.range(0, x.rank - 1))) {
|
return regularNormalizeBatchInTraining(x, gamma, beta, reductionAxes, epsilon);
|
}
|
else {
|
return broadcastNormalizeBatchInTraining(x, gamma, beta, reductionAxes, epsilon);
|
}
|
}
|
class BatchNormalization extends Layer {
|
constructor(args) {
|
if (args == null) {
|
args = {};
|
}
|
super(args);
|
this.supportsMasking = true;
|
this.axis = args.axis == null ? -1 : args.axis;
|
this.momentum = args.momentum == null ? 0.99 : args.momentum;
|
this.epsilon = args.epsilon == null ? 1e-3 : args.epsilon;
|
this.center = args.center == null ? true : args.center;
|
this.scale = args.scale == null ? true : args.scale;
|
this.betaInitializer = getInitializer(args.betaInitializer || 'zeros');
|
this.gammaInitializer = getInitializer(args.gammaInitializer || 'ones');
|
this.movingMeanInitializer =
|
getInitializer(args.movingMeanInitializer || 'zeros');
|
this.movingVarianceInitializer =
|
getInitializer(args.movingVarianceInitializer || 'ones');
|
this.betaConstraint = getConstraint(args.betaConstraint);
|
this.gammaConstraint = getConstraint(args.gammaConstraint);
|
this.betaRegularizer = getRegularizer(args.betaRegularizer);
|
this.gammaRegularizer = getRegularizer(args.gammaRegularizer);
|
}
|
build(inputShape) {
|
inputShape = getExactlyOneShape(inputShape);
|
const axis = this.axis >= 0 ? this.axis : (this.axis + inputShape.length);
|
const dim = inputShape[axis];
|
if (dim == null) {
|
throw new ValueError(`Axis ${axis} of input tensor should have a defined dimension but ` +
|
`the layer received an input with shape ` +
|
`${JSON.stringify(inputShape)}.`);
|
}
|
this.inputSpec =
|
[new InputSpec({ ndim: inputShape.length, axes: { [axis]: dim } })];
|
const shape = [dim];
|
if (this.scale) {
|
this.gamma = this.addWeight('gamma', shape, null, this.gammaInitializer, this.gammaRegularizer, true, this.gammaConstraint);
|
}
|
if (this.center) {
|
this.beta = this.addWeight('beta', shape, null, this.betaInitializer, this.betaRegularizer, true, this.betaConstraint);
|
}
|
this.movingMean = this.addWeight('moving_mean', shape, null, this.movingMeanInitializer, null, false);
|
this.movingVariance = this.addWeight('moving_variance', shape, null, this.movingVarianceInitializer, null, false);
|
this.built = true;
|
}
|
call(inputs, kwargs) {
|
return tidy(() => {
|
const training = kwargs['training'] == null ? false : kwargs['training'];
|
const input = getExactlyOneTensor(inputs);
|
const inputShape = input.shape;
|
const ndim = inputShape.length;
|
const reductionAxes = math_utils.range(0, ndim);
|
const axis = this.axis >= 0 ? this.axis : (this.axis + ndim);
|
reductionAxes.splice(axis, 1);
|
const broadcastShape = generic_utils.pyListRepeat(1, ndim);
|
broadcastShape[axis] = inputShape[axis];
|
const sortedReductionAxes = reductionAxes.slice();
|
sortedReductionAxes.sort();
|
const needsBroadcasting = !util.arraysEqual(sortedReductionAxes, math_utils.range(0, ndim).slice(0, ndim - 1));
|
const normalizeInference = () => {
|
if (needsBroadcasting) {
|
const broadcastMovingMean = reshape(this.movingMean.read(), broadcastShape);
|
const broadcastMovingVariance = reshape(this.movingVariance.read(), broadcastShape);
|
const broadcastBeta = this.center ? reshape(this.beta.read(), broadcastShape) : null;
|
const broadcastGamma = this.scale ? reshape(this.gamma.read(), broadcastShape) : null;
|
return batchNormalization(input, broadcastMovingMean, broadcastMovingVariance, broadcastBeta, broadcastGamma, this.epsilon);
|
}
|
else {
|
return batchNormalization(input, this.movingMean.read(), this.movingVariance.read(), this.beta == null ? null : this.beta.read(), this.gamma == null ? null : this.gamma.read(), this.epsilon);
|
}
|
};
|
if (!training) {
|
return normalizeInference();
|
}
|
const [normedTraining, mean, variance] = normalizeBatchInTraining(input, this.gamma.read(), this.beta.read(), reductionAxes, this.epsilon);
|
const doMovingAverage = (variable, value, momentum) => {
|
tfc.tidy(() => {
|
const decay = 1 - momentum;
|
const origValue = variable.read();
|
const updateDelta = tfc.mul(tfc.sub(origValue, value), decay);
|
variable.write(tfc.sub(origValue, updateDelta));
|
});
|
};
|
// Perform updates to moving mean and moving variance for training.
|
// Porting Note: In PyKeras, these updates to `movingMean` and
|
// `movingAverage` are done as a deferred Graph, added to the `Layer`'s
|
// `update`s using the `add_update()` method. Here we do it imperatively
|
// and encapsulate the updates in a function that is invoked
|
// immediately.
|
const updateMovingMeanAndVariance = () => {
|
doMovingAverage(this.movingMean, mean, this.momentum);
|
doMovingAverage(this.movingVariance, variance, this.momentum);
|
};
|
updateMovingMeanAndVariance();
|
return normedTraining;
|
});
|
}
|
getConfig() {
|
const config = {
|
axis: this.axis,
|
momentum: this.momentum,
|
epsilon: this.epsilon,
|
center: this.center,
|
scale: this.scale,
|
betaInitializer: serializeInitializer(this.betaInitializer),
|
gammaInitializer: serializeInitializer(this.gammaInitializer),
|
movingMeanInitializer: serializeInitializer(this.movingMeanInitializer),
|
movingVarianceInitializer: serializeInitializer(this.movingVarianceInitializer),
|
betaRegularizer: serializeRegularizer(this.betaRegularizer),
|
gammaRegularizer: serializeRegularizer(this.gammaRegularizer),
|
betaConstraint: serializeConstraint(this.betaConstraint),
|
gammaConstraint: serializeConstraint(this.gammaConstraint)
|
};
|
const baseConfig = super.getConfig();
|
Object.assign(config, baseConfig);
|
return config;
|
}
|
}
|
/** @nocollapse */
|
BatchNormalization.className = 'BatchNormalization';
|
export { BatchNormalization };
|
serialization.registerClass(BatchNormalization);
|
class LayerNormalization extends Layer {
|
constructor(args) {
|
if (args == null) {
|
args = {};
|
}
|
super(args);
|
this.axis = args.axis == null ? -1 : args.axis;
|
if (typeof this.axis === 'number') {
|
if (!Number.isInteger(this.axis)) {
|
throw new Error(`Expected axis to be an integer, but received ${this.axis}`);
|
}
|
}
|
else if (Array.isArray(this.axis)) {
|
for (const axis of this.axis) {
|
if (!Number.isInteger(axis)) {
|
throw new Error(`Expected axis to be an array of integers, ` +
|
`but received ${JSON.stringify(this.axis)}`);
|
}
|
}
|
}
|
else {
|
throw new Error(`Expected axis to be an integer or an array of integers, ` +
|
`but received ${JSON.stringify(this.axis)}`);
|
}
|
this.epsilon = args.epsilon == null ? 1e-3 : args.epsilon;
|
this.center = args.center == null ? true : args.center;
|
this.scale = args.scale == null ? true : args.scale;
|
this.betaInitializer = getInitializer(args.betaInitializer || 'zeros');
|
this.gammaInitializer = getInitializer(args.gammaInitializer || 'ones');
|
this.betaRegularizer = getRegularizer(args.betaRegularizer);
|
this.gammaRegularizer = getRegularizer(args.gammaRegularizer);
|
this.supportsMasking = true;
|
}
|
build(inputShape) {
|
inputShape = getExactlyOneShape(inputShape);
|
const nDims = inputShape.length;
|
// Convert axis to array and resolve negatives.
|
if (typeof this.axis === 'number') {
|
this.axis = [this.axis];
|
}
|
for (let i = 0; i < this.axis.length; ++i) {
|
if (this.axis[i] < 0) {
|
this.axis[i] += nDims;
|
}
|
}
|
// Further validate axes.
|
for (const axis of this.axis) {
|
if (axis < 0 || axis >= nDims) {
|
throw new Error(`Invalid axis: ${axis}`);
|
}
|
}
|
if (this.axis.length !== generic_utils.unique(this.axis).length) {
|
throw new Error(`Found duplicate axes in: ${this.axis}`);
|
}
|
const paramShape = this.axis.map(axis => inputShape[axis]);
|
const trainable = true;
|
if (this.scale) {
|
this.gamma = this.addWeight('gamma', paramShape, 'float32', this.gammaInitializer, this.gammaRegularizer, trainable);
|
}
|
else {
|
this.gamma = null;
|
}
|
if (this.center) {
|
this.beta = this.addWeight('beta', paramShape, 'float32', this.betaInitializer, this.betaRegularizer, trainable);
|
}
|
else {
|
this.beta = null;
|
}
|
this.built = true;
|
}
|
call(inputs, kwargs) {
|
const input = getExactlyOneTensor(inputs);
|
const inputShape = input.shape;
|
const nDims = inputShape.length;
|
return tidy(() => {
|
const keepDims = true;
|
let { mean, variance } = moments(input, this.axis, keepDims);
|
const broadcastShape = generic_utils.pyListRepeat(1, nDims);
|
for (const dim of this.axis) {
|
broadcastShape[dim] = inputShape[dim];
|
}
|
const broadcast = (v) => {
|
if (v != null && v.shape.length !== nDims) {
|
return tfc.reshape(v, broadcastShape);
|
}
|
else {
|
return v;
|
}
|
};
|
let scale = this.scale ? broadcast(this.gamma.read()) : null;
|
let offset = this.center ? broadcast(this.beta.read()) : null;
|
// TODO(https://github.com/tensorflow/tfjs/issues/2120): The tiling below
|
// is a workaround for the limitation of core's batchNormalization?d don't
|
// support broadcasting in their gradients. In addition, the tiling is
|
// necessary to ensure correctness on the browser CPU backend regardless
|
// of forward or backward computation. Remove this workaround once the
|
// limitation is addressed. See .
|
const momentsTiling = [];
|
const scaleOffsetTiling = [];
|
for (let i = 0; i < nDims; ++i) {
|
if (this.axis.indexOf(i) !== -1) {
|
momentsTiling.push(inputShape[i]);
|
scaleOffsetTiling.push(1);
|
}
|
else {
|
momentsTiling.push(1);
|
scaleOffsetTiling.push(inputShape[i]);
|
}
|
}
|
mean = tfc.tile(mean, momentsTiling);
|
variance = tfc.tile(variance, momentsTiling);
|
if (scale != null) {
|
scale = tfc.tile(scale, scaleOffsetTiling);
|
}
|
if (offset != null) {
|
offset = tfc.tile(offset, scaleOffsetTiling);
|
}
|
return batchNormalization(input, mean, variance, offset, scale, this.epsilon);
|
});
|
}
|
getConfig() {
|
const config = {
|
axis: this.axis,
|
epsilon: this.epsilon,
|
center: this.center,
|
scale: this.scale,
|
betaInitializer: serializeInitializer(this.betaInitializer),
|
gammaInitializer: serializeInitializer(this.gammaInitializer),
|
betaRegularizer: serializeRegularizer(this.betaRegularizer),
|
gammaRegularizer: serializeRegularizer(this.gammaRegularizer)
|
};
|
const baseConfig = super.getConfig();
|
Object.assign(config, baseConfig);
|
return config;
|
}
|
}
|
/** @nocollapse */
|
LayerNormalization.className = 'LayerNormalization';
|
export { LayerNormalization };
|
serialization.registerClass(LayerNormalization);
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"normalization.js","sourceRoot":"","sources":["../../../../../../tfjs-layers/src/layers/normalization.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright 2018 Google LLC\n *\n * Use of this source code is governed by an MIT-style\n * license that can be found in the LICENSE file or at\n * https://opensource.org/licenses/MIT.\n * =============================================================================\n */\n\n/**\n * Normalization layers.\n */\n\nimport * as tfc from '@tensorflow/tfjs-core';\nimport {moments, reshape, serialization, Tensor, Tensor1D, Tensor2D, Tensor3D, Tensor4D, tidy, util} from '@tensorflow/tfjs-core';\n\nimport {Constraint, ConstraintIdentifier, getConstraint, serializeConstraint} from '../constraints';\nimport {InputSpec, Layer, LayerArgs} from '../engine/topology';\nimport {NotImplementedError, ValueError} from '../errors';\nimport {getInitializer, Initializer, InitializerIdentifier, serializeInitializer} from '../initializers';\nimport {Shape} from '../keras_format/common';\nimport {getRegularizer, Regularizer, RegularizerIdentifier, serializeRegularizer} from '../regularizers';\nimport {Kwargs} from '../types';\nimport * as generic_utils from '../utils/generic_utils';\nimport * as math_utils from '../utils/math_utils';\nimport {getExactlyOneShape, getExactlyOneTensor} from '../utils/types_utils';\nimport {LayerVariable} from '../variables';\n\n/**\n * Applies batch normalization on x given mean, var, beta and gamma.\n *\n * I.e. returns:\n *   `output = (x - mean) / (sqrt(var) + epsilon) * gamma + beta`\n *\n * @param x Input tensor.\n * @param mean Mean of batch.\n * @param variance Variance of batch.\n * @param beta Tensor with which to center the input.\n * @param gamma Tensor by which to scale the input.\n * @param epsilon Fuzz factor.\n * @returns The result of the batch normalization.\n */\nexport function batchNormalization(\n    x: Tensor, mean: Tensor, variance: Tensor, beta?: Tensor, gamma?: Tensor,\n    epsilon = 1e-3): Tensor {\n  let out: Tensor;\n  if (x.rank === 2) {\n    out = tfc.batchNorm2d(\n        x as Tensor2D, mean as Tensor2D | Tensor1D,\n        variance as Tensor2D | Tensor1D, beta as Tensor2D | Tensor1D,\n        gamma as Tensor2D | Tensor1D, epsilon);\n  } else if (x.rank === 3) {\n    // TODO(cais): Check rank; give proper error message.\n    out = tfc.batchNorm3d(\n        x as Tensor3D, mean as Tensor3D | Tensor1D,\n        variance as Tensor3D | Tensor1D, beta as Tensor3D | Tensor1D,\n        gamma as Tensor3D | Tensor1D, epsilon);\n  } else if (x.rank === 4) {\n    out = tfc.batchNorm4d(\n        x as Tensor4D, mean as Tensor4D | Tensor1D,\n        variance as Tensor4D | Tensor1D, beta as Tensor4D | Tensor1D,\n        gamma as Tensor4D | Tensor1D, epsilon);\n  } else {\n    throw new NotImplementedError(\n        `batchNormalization is not implemented for array of rank ${x.rank} ` +\n        `yet`);\n  }\n  return out;\n}\n\n/**\n * Non-broadcasting batch normalization for use in training (not inference).\n *\n * The input is normalized to zero mean and unit variance along the\n * `reductionAxes`, followed by scaling with `gamma` and shifted by `beta`.\n * The result of that is returned as the first element\n * of the returned `Array`. The other two elements are the mean and variance,\n * respectively.\n *\n * @param x Input tensor to be normalized.\n * @param gamma Tensor by which to scale the input.\n * @param beta Tensor by which to center the input.\n * @param reductionAxes Axes over which to normalize.\n * @param epsilon Fuzz factor.\n * @returns An `Array` of three `Tensors`:\n *   [normalized tensor, mean of input, variance of input].\n */\nfunction regularNormalizeBatchInTraining(\n    x: Tensor, gamma: Tensor, beta: Tensor, reductionAxes: number[],\n    epsilon = 1e-3): [Tensor, Tensor, Tensor] {\n  return tidy(() => {\n           const meanAndVariance = tfc.moments(x, reductionAxes);\n           const mean = meanAndVariance.mean;\n           const variance = meanAndVariance.variance;\n           const normed =\n               batchNormalization(x, mean, variance, beta, gamma, epsilon);\n           return [normed, mean, variance];\n         }) as [Tensor, Tensor, Tensor];\n}\n\n/**\n * Broadcasting batch normalization for use in training (not inference).\n *\n * The input is normalized to zero mean and unit variance along the\n * `reductionAxes`, followed by scaling with `gamma` and shifted by `beta`.\n * The result of that is returned as the first element\n * of the returned `Array`. The other two elements are the mean and variance,\n * respectively.\n *\n * @param x Input tensor to be normalized.\n * @param gamma Tensor by which to scale the input.\n * @param beta Tensor by which to center the input.\n * @param reductionAxes Axes over which to normalize.\n * @param epsilon Fuzz factor.\n * @returns An `Array` of three `Tensors`:\n *   [normalized tensor, mean of input, variance of input].\n */\nfunction broadcastNormalizeBatchInTraining(\n    x: Tensor, gamma: Tensor, beta: Tensor, reductionAxes: number[],\n    epsilon = 1e-3): [Tensor, Tensor, Tensor] {\n  return tidy(() => {\n           const meanAndVariance = tfc.moments(x, reductionAxes);\n           const mean = meanAndVariance.mean;\n           const variance = meanAndVariance.variance;\n           const targetShape: number[] = [];\n           for (const axis of math_utils.range(0, x.rank)) {\n             if (reductionAxes.indexOf(axis) !== -1) {\n               targetShape.push(1);\n             } else {\n               targetShape.push(x.shape[axis]);\n             }\n           }\n           const broadcastMean = reshape(mean, targetShape);\n           const broadcastVariance = reshape(variance, targetShape);\n           const broadcastGamma =\n               gamma == null ? null : reshape(gamma, targetShape);\n           const broadcastBeta =\n               beta == null ? null : reshape(beta, targetShape);\n           const normed = batchNormalization(\n               x, broadcastMean, broadcastVariance, broadcastBeta,\n               broadcastGamma, epsilon);\n           return [normed, mean, variance];\n         }) as [Tensor, Tensor, Tensor];\n}\n\n/**\n * Batch normalization for use in training (not inference).\n *\n * @param x Input tensor to be normalized.\n * @param gamma Tensor by which to scale the input.\n * @param beta Tensor by which to center the input.\n * @param reductionAxes Axes over which to normalize.\n * @param epsilon Fuzz factor.\n * @returns An `Array` of three `Tensors`:\n *   [normalized tensor, mean of input, variance of input].\n */\nexport function normalizeBatchInTraining(\n    x: Tensor, gamma: Tensor, beta: Tensor, reductionAxes: number[],\n    epsilon = 1e-3): [Tensor, Tensor, Tensor] {\n  if (util.arraysEqual(\n          reductionAxes.slice().sort(), math_utils.range(0, x.rank - 1))) {\n    return regularNormalizeBatchInTraining(\n        x, gamma, beta, reductionAxes, epsilon);\n  } else {\n    return broadcastNormalizeBatchInTraining(\n        x, gamma, beta, reductionAxes, epsilon);\n  }\n}\n\nexport declare interface BatchNormalizationLayerArgs extends LayerArgs {\n  /**\n   * The integer axis that should be normalized (typically the features axis).\n   * Defaults to -1.\n   *\n   * For instance, after a `Conv2D` layer with `data_format=\"channels_first\"`,\n   * set `axis=1` in `batchNormalization`.\n   */\n  axis?: number;\n\n  /**\n   * Momentum of the moving average. Defaults to 0.99.\n   */\n  momentum?: number;\n\n  /**\n   * Small float added to the variance to avoid dividing by zero. Defaults to\n   * 1e-3.\n   */\n  epsilon?: number;\n\n  /**\n   * If `true`, add offset of `beta` to normalized tensor.\n   * If `false`, `beta` is ignored.\n   * Defaults to `true`.\n   */\n  center?: boolean;\n\n  /**\n   * If `true`, multiply by `gamma`.\n   * If `false`, `gamma` is not used.\n   * When the next layer is linear (also e.g. `nn.relu`),\n   * this can be disabled since the scaling will be done by the next layer.\n   * Defaults to `true`.\n   */\n  scale?: boolean;\n\n  /**\n   * Initializer for the beta weight.\n   *  Defaults to 'zeros'.\n   */\n  betaInitializer?: InitializerIdentifier|Initializer;\n\n  /**\n   * Initializer for the gamma weight.\n   *  Defaults to `ones`.\n   */\n  gammaInitializer?: InitializerIdentifier|Initializer;\n\n  /**\n   * Initializer for the moving mean.\n   * Defaults to `zeros`\n   */\n  movingMeanInitializer?: InitializerIdentifier|Initializer;\n\n  /**\n   * Initializer for the moving variance.\n   *  Defaults to 'Ones'.\n   */\n  movingVarianceInitializer?: InitializerIdentifier|Initializer;\n\n  /**\n   * Constraint for the beta weight.\n   */\n  betaConstraint?: ConstraintIdentifier|Constraint;\n\n  /**\n   * Constraint for gamma weight.\n   */\n  gammaConstraint?: ConstraintIdentifier|Constraint;\n\n  /**\n   * Regularizer for the beta weight.\n   */\n  betaRegularizer?: RegularizerIdentifier|Regularizer;\n\n  /**\n   * Regularizer for the gamma weight.\n   */\n  gammaRegularizer?: RegularizerIdentifier|Regularizer;\n}\n\nexport class BatchNormalization extends Layer {\n  /** @nocollapse */\n  static className = 'BatchNormalization';\n  private readonly axis: number;\n  private readonly momentum: number;\n  private readonly epsilon: number;\n  private readonly center: boolean;\n  private readonly scale: boolean;\n  private readonly betaInitializer: Initializer;\n  private readonly gammaInitializer: Initializer;\n  private readonly movingMeanInitializer: Initializer;\n  private readonly movingVarianceInitializer: Initializer;\n  private readonly betaConstraint: Constraint;\n  private readonly gammaConstraint: Constraint;\n  private readonly betaRegularizer: Regularizer;\n  private readonly gammaRegularizer: Regularizer;\n  private gamma: LayerVariable;\n  private beta: LayerVariable;\n  private movingMean: LayerVariable;\n  private movingVariance: LayerVariable;\n\n  constructor(args?: BatchNormalizationLayerArgs) {\n    if (args == null) {\n      args = {};\n    }\n    super(args);\n\n    this.supportsMasking = true;\n    this.axis = args.axis == null ? -1 : args.axis;\n    this.momentum = args.momentum == null ? 0.99 : args.momentum;\n    this.epsilon = args.epsilon == null ? 1e-3 : args.epsilon;\n    this.center = args.center == null ? true : args.center;\n    this.scale = args.scale == null ? true : args.scale;\n    this.betaInitializer = getInitializer(args.betaInitializer || 'zeros');\n    this.gammaInitializer = getInitializer(args.gammaInitializer || 'ones');\n    this.movingMeanInitializer =\n        getInitializer(args.movingMeanInitializer || 'zeros');\n    this.movingVarianceInitializer =\n        getInitializer(args.movingVarianceInitializer || 'ones');\n    this.betaConstraint = getConstraint(args.betaConstraint);\n    this.gammaConstraint = getConstraint(args.gammaConstraint);\n    this.betaRegularizer = getRegularizer(args.betaRegularizer);\n    this.gammaRegularizer = getRegularizer(args.gammaRegularizer);\n  }\n\n  public override build(inputShape: Shape|Shape[]): void {\n    inputShape = getExactlyOneShape(inputShape);\n    const axis = this.axis >= 0 ? this.axis : (this.axis + inputShape.length);\n    const dim = inputShape[axis];\n    if (dim == null) {\n      throw new ValueError(\n          `Axis ${axis} of input tensor should have a defined dimension but ` +\n          `the layer received an input with shape ` +\n          `${JSON.stringify(inputShape)}.`);\n    }\n    this.inputSpec =\n        [new InputSpec({ndim: inputShape.length, axes: {[axis]: dim}})];\n    const shape = [dim];\n    if (this.scale) {\n      this.gamma = this.addWeight(\n          'gamma', shape, null, this.gammaInitializer, this.gammaRegularizer,\n          true, this.gammaConstraint);\n    }\n    if (this.center) {\n      this.beta = this.addWeight(\n          'beta', shape, null, this.betaInitializer, this.betaRegularizer, true,\n          this.betaConstraint);\n    }\n    this.movingMean = this.addWeight(\n        'moving_mean', shape, null, this.movingMeanInitializer, null, false);\n    this.movingVariance = this.addWeight(\n        'moving_variance', shape, null, this.movingVarianceInitializer, null,\n        false);\n    this.built = true;\n  }\n\n  override call(inputs: Tensor|Tensor[], kwargs: Kwargs): Tensor|Tensor[] {\n    return tidy(() => {\n      const training = kwargs['training'] == null ? false : kwargs['training'];\n      const input = getExactlyOneTensor(inputs);\n      const inputShape = input.shape;\n      const ndim = inputShape.length;\n      const reductionAxes = math_utils.range(0, ndim);\n      const axis = this.axis >= 0 ? this.axis : (this.axis + ndim);\n      reductionAxes.splice(axis, 1);\n      const broadcastShape = generic_utils.pyListRepeat(1, ndim);\n      broadcastShape[axis] = inputShape[axis];\n\n      const sortedReductionAxes = reductionAxes.slice();\n      sortedReductionAxes.sort();\n      const needsBroadcasting = !util.arraysEqual(\n          sortedReductionAxes, math_utils.range(0, ndim).slice(0, ndim - 1));\n\n      const normalizeInference: () => Tensor = () => {\n        if (needsBroadcasting) {\n          const broadcastMovingMean =\n              reshape(this.movingMean.read(), broadcastShape);\n          const broadcastMovingVariance =\n              reshape(this.movingVariance.read(), broadcastShape);\n          const broadcastBeta =\n              this.center ? reshape(this.beta.read(), broadcastShape) : null;\n          const broadcastGamma =\n              this.scale ? reshape(this.gamma.read(), broadcastShape) : null;\n          return batchNormalization(\n              input, broadcastMovingMean, broadcastMovingVariance,\n              broadcastBeta, broadcastGamma, this.epsilon);\n        } else {\n          return batchNormalization(\n              input, this.movingMean.read(), this.movingVariance.read(),\n              this.beta == null ? null : this.beta.read(),\n              this.gamma == null ? null : this.gamma.read(), this.epsilon);\n        }\n      };\n\n      if (!training) {\n        return normalizeInference();\n      }\n\n      const [normedTraining, mean, variance] = normalizeBatchInTraining(\n          input, this.gamma.read(), this.beta.read(), reductionAxes,\n          this.epsilon);\n\n      const doMovingAverage =\n          (variable: LayerVariable, value: Tensor, momentum: number): void => {\n            tfc.tidy(() => {\n              const decay = 1 - momentum;\n              const origValue = variable.read();\n              const updateDelta = tfc.mul(tfc.sub(origValue, value), decay);\n              variable.write(tfc.sub(origValue, updateDelta));\n            });\n          };\n\n      // Perform updates to moving mean and moving variance for training.\n      // Porting Note: In PyKeras, these updates to `movingMean` and\n      //   `movingAverage` are done as a deferred Graph, added to the `Layer`'s\n      //   `update`s using the `add_update()` method. Here we do it imperatively\n      //   and encapsulate the updates in a function that is invoked\n      //   immediately.\n      const updateMovingMeanAndVariance = () => {\n        doMovingAverage(this.movingMean, mean, this.momentum);\n        doMovingAverage(this.movingVariance, variance, this.momentum);\n      };\n      updateMovingMeanAndVariance();\n\n      return normedTraining;\n    });\n  }\n\n  override getConfig(): serialization.ConfigDict {\n    const config: serialization.ConfigDict = {\n      axis: this.axis,\n      momentum: this.momentum,\n      epsilon: this.epsilon,\n      center: this.center,\n      scale: this.scale,\n      betaInitializer: serializeInitializer(this.betaInitializer),\n      gammaInitializer: serializeInitializer(this.gammaInitializer),\n      movingMeanInitializer: serializeInitializer(this.movingMeanInitializer),\n      movingVarianceInitializer:\n          serializeInitializer(this.movingVarianceInitializer),\n      betaRegularizer: serializeRegularizer(this.betaRegularizer),\n      gammaRegularizer: serializeRegularizer(this.gammaRegularizer),\n      betaConstraint: serializeConstraint(this.betaConstraint),\n      gammaConstraint: serializeConstraint(this.gammaConstraint)\n    };\n    const baseConfig = super.getConfig();\n    Object.assign(config, baseConfig);\n    return config;\n  }\n}\nserialization.registerClass(BatchNormalization);\n\nexport interface LayerNormalizationLayerArgs extends LayerArgs {\n  /**\n   * The axis or axes that should be normalized (typically, the feature axis).\n   * Defaults to -1 (the last axis).\n   */\n  axis?: number|number[];\n\n  /**\n   * A small positive float added to variance to avoid divison by zero.\n   * Defaults to 1e-3.\n   */\n  epsilon?: number;\n\n  /**\n   * If `true`, add offset of `beta` to normalized tensor.\n   * If `false`, `beta` is ignored.\n   * Default: `true`.\n   */\n  center?: boolean;\n\n  /**\n   * If `true`, multiply output by `gamma`.\n   * If `false`, `gamma` is not used.\n   * When the next layer is linear, this can be disabled since scaling will\n   * be done by the next layer.\n   * Default: `true`.\n   */\n  scale?: boolean;\n\n  /**\n   * Initializer for the beta weight.\n   * Default: `'zeros'`.\n   */\n  betaInitializer?: InitializerIdentifier|Initializer;\n\n  /**\n   * Initializer for the gamma weight.\n   * Default: `'ones'`.\n   */\n  gammaInitializer?: InitializerIdentifier|Initializer;\n\n  /** Regularizer for the beta weight. */\n  betaRegularizer?: RegularizerIdentifier|Regularizer;\n\n  /** Regularizer for the gamma weight. */\n  gammaRegularizer?: RegularizerIdentifier|Regularizer;\n}\n\nexport class LayerNormalization extends Layer {\n  /** @nocollapse */\n  static className = 'LayerNormalization';\n\n  private axis: number|number[];\n  readonly epsilon: number;\n  readonly center: boolean;\n  readonly scale: boolean;\n  readonly betaInitializer: Initializer;\n  readonly gammaInitializer: Initializer;\n  readonly betaRegularizer: Regularizer;\n  readonly gammaRegularizer: Regularizer;\n\n  private gamma: LayerVariable;\n  private beta: LayerVariable;\n\n  constructor(args?: LayerNormalizationLayerArgs) {\n    if (args == null) {\n      args = {};\n    }\n    super(args);\n\n    this.axis = args.axis == null ? -1 : args.axis;\n    if (typeof this.axis === 'number') {\n      if (!Number.isInteger(this.axis)) {\n        throw new Error(\n            `Expected axis to be an integer, but received ${this.axis}`);\n      }\n    } else if (Array.isArray(this.axis)) {\n      for (const axis of this.axis) {\n        if (!Number.isInteger(axis)) {\n          throw new Error(\n              `Expected axis to be an array of integers, ` +\n              `but received ${JSON.stringify(this.axis)}`);\n        }\n      }\n    } else {\n      throw new Error(\n          `Expected axis to be an integer or an array of integers, ` +\n          `but received ${JSON.stringify(this.axis)}`);\n    }\n\n    this.epsilon = args.epsilon == null ? 1e-3 : args.epsilon;\n    this.center = args.center == null ? true : args.center;\n    this.scale = args.scale == null ? true : args.scale;\n    this.betaInitializer = getInitializer(args.betaInitializer || 'zeros');\n    this.gammaInitializer = getInitializer(args.gammaInitializer || 'ones');\n    this.betaRegularizer = getRegularizer(args.betaRegularizer);\n    this.gammaRegularizer = getRegularizer(args.gammaRegularizer);\n\n    this.supportsMasking = true;\n  }\n\n  public override build(inputShape: Shape|Shape[]): void {\n    inputShape = getExactlyOneShape(inputShape);\n    const nDims = inputShape.length;\n\n    // Convert axis to array and resolve negatives.\n    if (typeof this.axis === 'number') {\n      this.axis = [this.axis];\n    }\n    for (let i = 0; i < this.axis.length; ++i) {\n      if (this.axis[i] < 0) {\n        this.axis[i] += nDims;\n      }\n    }\n\n    // Further validate axes.\n    for (const axis of this.axis) {\n      if (axis < 0 || axis >= nDims) {\n        throw new Error(`Invalid axis: ${axis}`);\n      }\n    }\n    if (this.axis.length !== generic_utils.unique(this.axis).length) {\n      throw new Error(`Found duplicate axes in: ${this.axis}`);\n    }\n\n    const paramShape = this.axis.map(axis => inputShape[axis]) as number[];\n\n    const trainable = true;\n    if (this.scale) {\n      this.gamma = this.addWeight(\n          'gamma', paramShape, 'float32', this.gammaInitializer,\n          this.gammaRegularizer, trainable);\n    } else {\n      this.gamma = null;\n    }\n    if (this.center) {\n      this.beta = this.addWeight(\n          'beta', paramShape, 'float32', this.betaInitializer,\n          this.betaRegularizer, trainable);\n    } else {\n      this.beta = null;\n    }\n\n    this.built = true;\n  }\n\n  override call(inputs: Tensor|Tensor[], kwargs: Kwargs): Tensor|Tensor[] {\n    const input = getExactlyOneTensor(inputs);\n    const inputShape = input.shape;\n    const nDims = inputShape.length;\n\n    return tidy(() => {\n      const keepDims = true;\n      let {mean, variance} = moments(input, this.axis, keepDims);\n      const broadcastShape = generic_utils.pyListRepeat(1, nDims);\n      for (const dim of this.axis as number[]) {\n        broadcastShape[dim] = inputShape[dim];\n      }\n\n      const broadcast = (v: Tensor) => {\n        if (v != null && v.shape.length !== nDims) {\n          return tfc.reshape(v, broadcastShape);\n        } else {\n          return v;\n        }\n      };\n\n      let scale = this.scale ? broadcast(this.gamma.read()) : null;\n      let offset = this.center ? broadcast(this.beta.read()) : null;\n\n      // TODO(https://github.com/tensorflow/tfjs/issues/2120): The tiling below\n      // is a workaround for the limitation of core's batchNormalization?d don't\n      // support broadcasting in their gradients. In addition, the tiling is\n      // necessary to ensure correctness on the browser CPU backend regardless\n      // of forward or backward computation. Remove this workaround once the\n      // limitation is addressed. See .\n      const momentsTiling: number[] = [];\n      const scaleOffsetTiling: number[] = [];\n      for (let i = 0; i < nDims; ++i) {\n        if ((this.axis as number[]).indexOf(i) !== -1) {\n          momentsTiling.push(inputShape[i]);\n          scaleOffsetTiling.push(1);\n        } else {\n          momentsTiling.push(1);\n          scaleOffsetTiling.push(inputShape[i]);\n        }\n      }\n      mean = tfc.tile(mean, momentsTiling);\n      variance = tfc.tile(variance, momentsTiling);\n      if (scale != null) {\n        scale = tfc.tile(scale, scaleOffsetTiling);\n      }\n      if (offset != null) {\n        offset = tfc.tile(offset, scaleOffsetTiling);\n      }\n\n      return batchNormalization(\n          input, mean, variance, offset, scale, this.epsilon);\n    });\n  }\n\n  override getConfig(): serialization.ConfigDict {\n    const config: serialization.ConfigDict = {\n      axis: this.axis,\n      epsilon: this.epsilon,\n      center: this.center,\n      scale: this.scale,\n      betaInitializer: serializeInitializer(this.betaInitializer),\n      gammaInitializer: serializeInitializer(this.gammaInitializer),\n      betaRegularizer: serializeRegularizer(this.betaRegularizer),\n      gammaRegularizer: serializeRegularizer(this.gammaRegularizer)\n    };\n    const baseConfig = super.getConfig();\n    Object.assign(config, baseConfig);\n    return config;\n  }\n}\nserialization.registerClass(LayerNormalization);\n"]}
|