提交 a4d54b83 编写于 作者: G guosheng

Make GRU Operator adapt to the latest code

上级 9162629b
...@@ -43,14 +43,12 @@ class GRUOp : public framework::OperatorWithKernel { ...@@ -43,14 +43,12 @@ class GRUOp : public framework::OperatorWithKernel {
PADDLE_ENFORCE_EQ( PADDLE_ENFORCE_EQ(
weight_dims[1], frame_size * 3, weight_dims[1], frame_size * 3,
"The shape of Weight matrix must be [frame_size, frame_size * 3]."); "The shape of Weight matrix must be [frame_size, frame_size * 3].");
auto h0 = Input("H0"); if (ctx->HasInput("H0")) {
if (h0 != framework::kEmptyVarName) {
auto h0_dims = ctx->GetInputDim("H0"); auto h0_dims = ctx->GetInputDim("H0");
PADDLE_ENFORCE_EQ(h0_dims[1], frame_size, PADDLE_ENFORCE_EQ(h0_dims[1], frame_size,
"The width of H0 must be equal to frame_size."); "The width of H0 must be equal to frame_size.");
} }
auto bias = Input("Bias"); if (ctx->HasInput("Bias")) {
if (bias != framework::kEmptyVarName) {
auto bias_dims = ctx->GetInputDim("Bias"); auto bias_dims = ctx->GetInputDim("Bias");
int bias_height = bias_dims[0]; int bias_height = bias_dims[0];
int bias_width = bias_dims[1]; int bias_width = bias_dims[1];
...@@ -74,42 +72,52 @@ class GRUOpMaker : public framework::OpProtoAndCheckerMaker { ...@@ -74,42 +72,52 @@ class GRUOpMaker : public framework::OpProtoAndCheckerMaker {
GRUOpMaker(framework::OpProto* proto, framework::OpAttrChecker* op_checker) GRUOpMaker(framework::OpProto* proto, framework::OpAttrChecker* op_checker)
: OpProtoAndCheckerMaker(proto, op_checker) { : OpProtoAndCheckerMaker(proto, op_checker) {
AddInput("Input", AddInput("Input",
"(LoDTensor) the first input is a LodTensor, which support " "(LoDTensor) The first input is a LodTensor, which support "
"variable-time length input sequence. The underlying tensor in " "variable-time length input sequence. The underlying tensor in "
"this LoDTenosr is a matrix with shape (T X 3D), where, T is the " "this LoDTenosr is a matrix with shape (T X 3D), where, T is the "
"total time steps in this mini-batch, D is the hidden size."); "total time steps in this mini-batch, D is the hidden size.");
AddInput("H0", AddInput("H0",
"(Tensor, optional) the initial hidden state is an optional " "(Tensor, optional) The initial hidden state is an optional "
"input. This is a tensor with shape (N x D), where N is the " "input. This is a tensor with shape (N x D), where N is the "
"batch size, D is the hidden size."); "batch size, D is the hidden size.")
.AsDispensable();
AddInput( AddInput(
"Weight", "Weight",
"(Tensor) Weight matrix with shape [hidden_size, hidden_size * 3]. " "(Tensor) The learnable hidden-hidden weight matrix with shape "
"The elements continuous in memory can be divided into two parts. " "(D x 3D), where D is the hidden size. The elements continuous in "
"The first part are weights of the update gate and reset gate " "memory can be divided into two parts. The first part are weights of "
"with shape [hidden_size, hidden_size * 2], and the second part are " "the update gate and reset gate with shape (D x 2D), and the second "
"weights of output candidate with shape [hidden_size, hidden_size]"); "part are weights of output candidate with shape (D x D).");
AddInput("Bias", AddInput("Bias",
"(Tensor) Bias vector with shape [1, hidden_size * 3] concating " "(Tensor, optional) Bias vector with shape (1 x 3D) concating "
"bias of the update gate, reset gate and output candidate."); "bias of the update gate, reset gate and output candidate.")
.AsDispensable();
AddOutput("BatchGate", AddOutput("BatchGate",
"(LoDTensor) the update gata, reset gate and output candidate " "(LoDTensor) To compute with batches, sequence data will be "
"lod tensor of GRU operator. " "reorganized into several successive batches each containing "
"The shape and lod is the same with the `Input`.") "data from the same time step. The LoDTensor BatchGate contains "
"the update gate, reset gate and output candidate values "
"organized in batches. The LoD size is 2. The first LoD contains "
"the batch offsets and the second LoD contains the indexes in "
"the raw sequence data.")
.AsIntermediate(); .AsIntermediate();
AddOutput( AddOutput(
"BatchResetHiddenPrev", "BatchResetHiddenPrev",
"(LoDTensor) the reseted hidden state lod tensor of GRU operator. " "(LoDTensor) The reseted hidden state LoDTensor organized in batches. "
"The shape and lod is the same with the `Input`.") "This LoDTensor is a matrix with shape (T X D) and has the same LoD "
"with `BatchGate`.")
.AsIntermediate(); .AsIntermediate();
AddOutput( AddOutput(
"BatchHidden", "BatchHidden",
"(LoDTensor) the reseted hidden state lod tensor of GRU operator. " "(LoDTensor) The hidden state LoDTensor organized in batches. "
"The shape and lod is the same with the `Input`.") "This LoDTensor is a matrix with shape (T X D) and has the same LoD "
"with `BatchGate`.")
.AsIntermediate(); .AsIntermediate();
AddOutput("Hidden", AddOutput(
"(LoDTensor) the hidden state lod tensor of GRU operator. " "Hidden",
"The shape and lod is the same with the `Input`."); "(LoDTensor) the hidden state LoDTensor organized in sequences. "
"This LoDTensor is a matrix with shape (T X D) and has the same LoD "
"with `BatchGate`.");
AddAttr<std::string>("activation", AddAttr<std::string>("activation",
"(string, default tanh) " "(string, default tanh) "
"The activation type used for output candidate {h}_t.") "The activation type used for output candidate {h}_t.")
...@@ -124,14 +132,14 @@ class GRUOpMaker : public framework::OpProtoAndCheckerMaker { ...@@ -124,14 +132,14 @@ class GRUOpMaker : public framework::OpProtoAndCheckerMaker {
"whether to compute reversed GRU.") "whether to compute reversed GRU.")
.SetDefault(false); .SetDefault(false);
AddComment(R"DOC( AddComment(R"DOC(
GRUOp implements part calculations of the GRU unit as following: GRUOp implements part calculations of the GRU as following:
\f[ \f[
update \ gate: u_t = actGate(xu_t + W_u * hidden_prev + bias_u) \\ update \ gate: u_t = actGate(xu_t + W_u * hidden_prev + bias_u) \\
reset \ gate: r_t = actGate(xr_t + W_r * hidden_prev + bias_r) \\ reset \ gate: r_t = actGate(xr_t + W_r * hidden_prev + bias_r) \\
output \ candidate: {h}_t = actNode(xc_t + W_c * dot(r_t, hidden_prev) + bias_c) \\ output \ candidate: {h}_t = actNode(xc_t + W_c * dot(r_t, hidden_prev) + bias_c) \\
output: h_t = dot((1-u_t), hidden_prev) + dot(u_t, {h}_t) output: h_t = dot((1-u_t), hidden_prev) + dot(u_t, {h}_t)
\f] \f]
The rest of GRU unit can be completed by using FCOp's output as the input of GRUOp. The rest of GRU can be completed by using FCOp's output as the input of GRUOp.
)DOC"); )DOC");
} }
}; };
...@@ -170,8 +178,7 @@ class GRUGradOp : public framework::OperatorWithKernel { ...@@ -170,8 +178,7 @@ class GRUGradOp : public framework::OperatorWithKernel {
PADDLE_ENFORCE_EQ( PADDLE_ENFORCE_EQ(
weight_width, frame_size * 3, weight_width, frame_size * 3,
"The shape of Weight matrix must be [frame_size, frame_size * 3]."); "The shape of Weight matrix must be [frame_size, frame_size * 3].");
auto h0 = Input("H0"); if (ctx->HasInput("H0")) {
if (h0 != framework::kEmptyVarName) {
auto h0_dims = ctx->GetInputDim("H0"); auto h0_dims = ctx->GetInputDim("H0");
PADDLE_ENFORCE_EQ(h0_dims[1], frame_size, PADDLE_ENFORCE_EQ(h0_dims[1], frame_size,
"The width of H0 must be equal to frame_size."); "The width of H0 must be equal to frame_size.");
...@@ -179,8 +186,7 @@ class GRUGradOp : public framework::OperatorWithKernel { ...@@ -179,8 +186,7 @@ class GRUGradOp : public framework::OperatorWithKernel {
if (ctx->HasOutput(h0_grad_name)) if (ctx->HasOutput(h0_grad_name))
ctx->SetOutputDim(h0_grad_name, h0_dims); ctx->SetOutputDim(h0_grad_name, h0_dims);
} }
auto bias = Input("Bias"); if (ctx->HasInput("Bias")) {
if (bias != framework::kEmptyVarName) {
auto bias_dims = ctx->GetInputDim("Bias"); auto bias_dims = ctx->GetInputDim("Bias");
int bias_height = bias_dims[0]; int bias_height = bias_dims[0];
int bias_width = bias_dims[1]; int bias_width = bias_dims[1];
......
...@@ -62,7 +62,7 @@ class TestGRUOp(OpTest): ...@@ -62,7 +62,7 @@ class TestGRUOp(OpTest):
return idx_in_seq_list return idx_in_seq_list
def gru_step(self, x, h_p, w, b): def gru_step(self, x, h_p, w, b):
print x.shape, h_p.shape, w.shape, b.shape # print x.shape, h_p.shape, w.shape, b.shape
batch_size = x.shape[0] batch_size = x.shape[0]
frame_size = w.shape[0] frame_size = w.shape[0]
g = x + np.tile(b, (batch_size, 1)) g = x + np.tile(b, (batch_size, 1))
...@@ -96,7 +96,7 @@ class TestGRUOp(OpTest): ...@@ -96,7 +96,7 @@ class TestGRUOp(OpTest):
num_batch = len(idx_in_seq_list) num_batch = len(idx_in_seq_list)
end_idx = 0 end_idx = 0
for batch_idx in range(num_batch): for batch_idx in range(num_batch):
print idx_in_seq_list[batch_idx] # print idx_in_seq_list[batch_idx]
x = input[idx_in_seq_list[batch_idx]] x = input[idx_in_seq_list[batch_idx]]
g, r_h_p, h = self.gru_step(x, h_p, w, b) g, r_h_p, h = self.gru_step(x, h_p, w, b)
if batch_idx < (num_batch - 1): if batch_idx < (num_batch - 1):
...@@ -112,7 +112,7 @@ class TestGRUOp(OpTest): ...@@ -112,7 +112,7 @@ class TestGRUOp(OpTest):
def set_data(self): def set_data(self):
lod = [[0, 2, 6, 9]] #[[0, 1, 2, 3]] lod = [[0, 2, 6, 9]] #[[0, 1, 2, 3]]
self.idx_in_seq_list = self.seq_to_batch(lod, self.is_reverse) self.idx_in_seq_list = self.seq_to_batch(lod, self.is_reverse)
print self.idx_in_seq_list # print self.idx_in_seq_list
batch_size = self.batch_size batch_size = self.batch_size
frame_size = self.frame_size frame_size = self.frame_size
input = np.random.rand(batch_size, frame_size * 3).astype('float64') input = np.random.rand(batch_size, frame_size * 3).astype('float64')
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册