2014-10-02 11:38:22 +08:00
|
|
|
// This file is part of Eigen, a lightweight C++ template library
|
|
|
|
// for linear algebra.
|
|
|
|
//
|
|
|
|
// Copyright (C) 2014 Benoit Steiner <benoit.steiner.goog@gmail.com>
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla
|
|
|
|
// Public License v. 2.0. If a copy of the MPL was not distributed
|
|
|
|
// with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
|
|
|
#include "main.h"
|
|
|
|
#include <limits>
|
|
|
|
#include <Eigen/CXX11/Tensor>
|
|
|
|
|
|
|
|
using Eigen::Tensor;
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
template <int DataLayout>
|
|
|
|
static void test_simple_reductions() {
|
|
|
|
Tensor<float, 4, DataLayout> tensor(2, 3, 5, 7);
|
2014-10-02 11:38:22 +08:00
|
|
|
tensor.setRandom();
|
|
|
|
array<ptrdiff_t, 2> reduction_axis;
|
|
|
|
reduction_axis[0] = 1;
|
|
|
|
reduction_axis[1] = 3;
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
Tensor<float, 2, DataLayout> result = tensor.sum(reduction_axis);
|
2014-10-02 11:38:22 +08:00
|
|
|
VERIFY_IS_EQUAL(result.dimension(0), 2);
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(1), 5);
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
|
|
for (int j = 0; j < 5; ++j) {
|
|
|
|
float sum = 0.0f;
|
|
|
|
for (int k = 0; k < 3; ++k) {
|
|
|
|
for (int l = 0; l < 7; ++l) {
|
|
|
|
sum += tensor(i, k, j, l);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(result(i, j), sum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
{
|
|
|
|
Tensor<float, 1, DataLayout> sum1 = tensor.sum();
|
|
|
|
VERIFY_IS_EQUAL(sum1.dimension(0), 1);
|
|
|
|
|
|
|
|
array<ptrdiff_t, 4> reduction_axis;
|
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 1;
|
|
|
|
reduction_axis[2] = 2;
|
|
|
|
reduction_axis[3] = 3;
|
|
|
|
Tensor<float, 1, DataLayout> sum2 = tensor.sum(reduction_axis);
|
|
|
|
VERIFY_IS_EQUAL(sum2.dimension(0), 1);
|
|
|
|
|
|
|
|
VERIFY_IS_APPROX(sum1(0), sum2(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 2;
|
|
|
|
result = tensor.prod(reduction_axis);
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(0), 3);
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(1), 7);
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
for (int j = 0; j < 7; ++j) {
|
|
|
|
float prod = 1.0f;
|
|
|
|
for (int k = 0; k < 2; ++k) {
|
|
|
|
for (int l = 0; l < 5; ++l) {
|
|
|
|
prod *= tensor(k, i, l, j);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(result(i, j), prod);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Tensor<float, 1, DataLayout> prod1 = tensor.prod();
|
|
|
|
VERIFY_IS_EQUAL(prod1.dimension(0), 1);
|
|
|
|
|
|
|
|
array<ptrdiff_t, 4> reduction_axis;
|
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 1;
|
|
|
|
reduction_axis[2] = 2;
|
|
|
|
reduction_axis[3] = 3;
|
|
|
|
Tensor<float, 1, DataLayout> prod2 = tensor.prod(reduction_axis);
|
|
|
|
VERIFY_IS_EQUAL(prod2.dimension(0), 1);
|
|
|
|
|
|
|
|
VERIFY_IS_APPROX(prod1(0), prod2(0));
|
|
|
|
}
|
|
|
|
|
2014-10-02 11:38:22 +08:00
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 2;
|
|
|
|
result = tensor.maximum(reduction_axis);
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(0), 3);
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(1), 7);
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
for (int j = 0; j < 7; ++j) {
|
|
|
|
float max_val = std::numeric_limits<float>::lowest();
|
|
|
|
for (int k = 0; k < 2; ++k) {
|
|
|
|
for (int l = 0; l < 5; ++l) {
|
|
|
|
max_val = (std::max)(max_val, tensor(k, i, l, j));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(result(i, j), max_val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
{
|
|
|
|
Tensor<float, 1, DataLayout> max1 = tensor.maximum();
|
|
|
|
VERIFY_IS_EQUAL(max1.dimension(0), 1);
|
|
|
|
|
|
|
|
array<ptrdiff_t, 4> reduction_axis;
|
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 1;
|
|
|
|
reduction_axis[2] = 2;
|
|
|
|
reduction_axis[3] = 3;
|
|
|
|
Tensor<float, 1, DataLayout> max2 = tensor.maximum(reduction_axis);
|
|
|
|
VERIFY_IS_EQUAL(max2.dimension(0), 1);
|
|
|
|
|
|
|
|
VERIFY_IS_APPROX(max1(0), max2(0));
|
|
|
|
}
|
|
|
|
|
2014-10-02 11:38:22 +08:00
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 1;
|
|
|
|
result = tensor.minimum(reduction_axis);
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(0), 5);
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(1), 7);
|
|
|
|
for (int i = 0; i < 5; ++i) {
|
|
|
|
for (int j = 0; j < 7; ++j) {
|
|
|
|
float min_val = (std::numeric_limits<float>::max)();
|
|
|
|
for (int k = 0; k < 2; ++k) {
|
|
|
|
for (int l = 0; l < 3; ++l) {
|
2015-01-15 07:46:04 +08:00
|
|
|
min_val = (std::min)(min_val, tensor(k, l, i, j));
|
2014-10-02 11:38:22 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(result(i, j), min_val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
{
|
|
|
|
Tensor<float, 1, DataLayout> min1 = tensor.minimum();
|
|
|
|
VERIFY_IS_EQUAL(min1.dimension(0), 1);
|
|
|
|
|
|
|
|
array<ptrdiff_t, 4> reduction_axis;
|
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 1;
|
|
|
|
reduction_axis[2] = 2;
|
|
|
|
reduction_axis[3] = 3;
|
|
|
|
Tensor<float, 1, DataLayout> min2 = tensor.minimum(reduction_axis);
|
|
|
|
VERIFY_IS_EQUAL(min2.dimension(0), 1);
|
2014-10-02 11:38:22 +08:00
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
VERIFY_IS_APPROX(min1(0), min2(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 1;
|
|
|
|
result = tensor.mean(reduction_axis);
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(0), 5);
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(1), 7);
|
|
|
|
for (int i = 0; i < 5; ++i) {
|
|
|
|
for (int j = 0; j < 7; ++j) {
|
|
|
|
float sum = 0.0f;
|
|
|
|
int count = 0;
|
|
|
|
for (int k = 0; k < 2; ++k) {
|
|
|
|
for (int l = 0; l < 3; ++l) {
|
|
|
|
sum += tensor(k, l, i, j);
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(result(i, j), sum / count);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Tensor<float, 1, DataLayout> mean1 = tensor.mean();
|
|
|
|
VERIFY_IS_EQUAL(mean1.dimension(0), 1);
|
|
|
|
|
|
|
|
array<ptrdiff_t, 4> reduction_axis;
|
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 1;
|
|
|
|
reduction_axis[2] = 2;
|
|
|
|
reduction_axis[3] = 3;
|
|
|
|
Tensor<float, 1, DataLayout> mean2 = tensor.mean(reduction_axis);
|
|
|
|
VERIFY_IS_EQUAL(mean2.dimension(0), 1);
|
|
|
|
|
|
|
|
VERIFY_IS_APPROX(mean1(0), mean2(0));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <int DataLayout>
|
|
|
|
static void test_full_reductions() {
|
|
|
|
Tensor<float, 2, DataLayout> tensor(2, 3);
|
2014-10-02 11:38:22 +08:00
|
|
|
tensor.setRandom();
|
|
|
|
array<ptrdiff_t, 2> reduction_axis;
|
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 1;
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
Tensor<float, 1, DataLayout> result = tensor.sum(reduction_axis);
|
2014-10-02 11:38:22 +08:00
|
|
|
VERIFY_IS_EQUAL(result.dimension(0), 1);
|
|
|
|
|
|
|
|
float sum = 0.0f;
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
|
|
for (int j = 0; j < 3; ++j) {
|
|
|
|
sum += tensor(i, j);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(result(0), sum);
|
|
|
|
|
|
|
|
result = tensor.square().sum(reduction_axis).sqrt();
|
|
|
|
VERIFY_IS_EQUAL(result.dimension(0), 1);
|
|
|
|
|
|
|
|
sum = 0.0f;
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
|
|
for (int j = 0; j < 3; ++j) {
|
|
|
|
sum += tensor(i, j) * tensor(i, j);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(result(0), sqrtf(sum));
|
|
|
|
}
|
|
|
|
|
|
|
|
struct UserReducer {
|
2015-01-15 07:46:04 +08:00
|
|
|
static const bool PacketAccess = false;
|
|
|
|
UserReducer(float offset) : offset_(offset) {}
|
|
|
|
void reduce(const float val, float* accum) { *accum += val * val; }
|
|
|
|
float initialize() const { return 0; }
|
|
|
|
float finalize(const float accum) const { return 1.0f / (accum + offset_); }
|
2014-10-02 11:38:22 +08:00
|
|
|
|
|
|
|
private:
|
2015-01-15 07:46:04 +08:00
|
|
|
const float offset_;
|
2014-10-02 11:38:22 +08:00
|
|
|
};
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
template <int DataLayout>
|
|
|
|
static void test_user_defined_reductions() {
|
|
|
|
Tensor<float, 2, DataLayout> tensor(5, 7);
|
2014-10-02 11:38:22 +08:00
|
|
|
tensor.setRandom();
|
|
|
|
array<ptrdiff_t, 1> reduction_axis;
|
|
|
|
reduction_axis[0] = 1;
|
|
|
|
|
|
|
|
UserReducer reducer(10.0f);
|
2015-01-15 07:46:04 +08:00
|
|
|
Tensor<float, 1, DataLayout> result = tensor.reduce(reduction_axis, reducer);
|
2014-10-02 11:38:22 +08:00
|
|
|
VERIFY_IS_EQUAL(result.dimension(0), 5);
|
|
|
|
for (int i = 0; i < 5; ++i) {
|
|
|
|
float expected = 10.0f;
|
|
|
|
for (int j = 0; j < 7; ++j) {
|
|
|
|
expected += tensor(i, j) * tensor(i, j);
|
|
|
|
}
|
|
|
|
expected = 1.0f / expected;
|
|
|
|
VERIFY_IS_APPROX(result(i), expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
template <int DataLayout>
|
|
|
|
static void test_tensor_maps() {
|
|
|
|
int inputs[2 * 3 * 5 * 7];
|
|
|
|
TensorMap<Tensor<int, 4, DataLayout> > tensor_map(inputs, 2, 3, 5, 7);
|
|
|
|
TensorMap<Tensor<const int, 4, DataLayout> > tensor_map_const(inputs, 2, 3, 5,
|
|
|
|
7);
|
|
|
|
const TensorMap<Tensor<const int, 4, DataLayout> > tensor_map_const_const(
|
|
|
|
inputs, 2, 3, 5, 7);
|
2014-10-10 06:36:23 +08:00
|
|
|
|
|
|
|
tensor_map.setRandom();
|
|
|
|
array<ptrdiff_t, 2> reduction_axis;
|
|
|
|
reduction_axis[0] = 1;
|
|
|
|
reduction_axis[1] = 3;
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
Tensor<int, 2, DataLayout> result = tensor_map.sum(reduction_axis);
|
|
|
|
Tensor<int, 2, DataLayout> result2 = tensor_map_const.sum(reduction_axis);
|
|
|
|
Tensor<int, 2, DataLayout> result3 =
|
|
|
|
tensor_map_const_const.sum(reduction_axis);
|
2014-10-10 06:36:23 +08:00
|
|
|
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
|
|
for (int j = 0; j < 5; ++j) {
|
|
|
|
int sum = 0;
|
|
|
|
for (int k = 0; k < 3; ++k) {
|
|
|
|
for (int l = 0; l < 7; ++l) {
|
|
|
|
sum += tensor_map(i, k, j, l);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_EQUAL(result(i, j), sum);
|
|
|
|
VERIFY_IS_EQUAL(result2(i, j), sum);
|
|
|
|
VERIFY_IS_EQUAL(result3(i, j), sum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
template <int DataLayout>
|
|
|
|
static void test_static_dims() {
|
|
|
|
Tensor<float, 4, DataLayout> in(72, 53, 97, 113);
|
|
|
|
Tensor<float, 2, DataLayout> out(72, 97);
|
|
|
|
in.setRandom();
|
|
|
|
|
2015-02-06 18:51:59 +08:00
|
|
|
#ifndef EIGEN_HAS_CONSTEXPR
|
2015-01-15 07:46:04 +08:00
|
|
|
array<int, 2> reduction_axis;
|
|
|
|
reduction_axis[0] = 1;
|
|
|
|
reduction_axis[1] = 3;
|
|
|
|
#else
|
|
|
|
Eigen::IndexList<Eigen::type2index<1>, Eigen::type2index<3> > reduction_axis;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
out = in.maximum(reduction_axis);
|
|
|
|
|
|
|
|
for (int i = 0; i < 72; ++i) {
|
|
|
|
for (int j = 0; j < 97; ++j) {
|
|
|
|
float expected = -1e10f;
|
|
|
|
for (int k = 0; k < 53; ++k) {
|
|
|
|
for (int l = 0; l < 113; ++l) {
|
|
|
|
expected = (std::max)(expected, in(i, k, j, l));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(out(i, j), expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <int DataLayout>
|
|
|
|
static void test_innermost_last_dims() {
|
|
|
|
Tensor<float, 4, DataLayout> in(72, 53, 97, 113);
|
|
|
|
Tensor<float, 2, DataLayout> out(97, 113);
|
|
|
|
in.setRandom();
|
|
|
|
|
|
|
|
// Reduce on the innermost dimensions.
|
2015-02-06 18:51:59 +08:00
|
|
|
#ifndef EIGEN_HAS_CONSTEXPR
|
2015-01-15 07:46:04 +08:00
|
|
|
array<int, 2> reduction_axis;
|
|
|
|
reduction_axis[0] = 0;
|
|
|
|
reduction_axis[1] = 1;
|
|
|
|
#else
|
|
|
|
// This triggers the use of packets for ColMajor.
|
|
|
|
Eigen::IndexList<Eigen::type2index<0>, Eigen::type2index<1> > reduction_axis;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
out = in.maximum(reduction_axis);
|
|
|
|
|
|
|
|
for (int i = 0; i < 97; ++i) {
|
|
|
|
for (int j = 0; j < 113; ++j) {
|
|
|
|
float expected = -1e10f;
|
|
|
|
for (int k = 0; k < 53; ++k) {
|
|
|
|
for (int l = 0; l < 72; ++l) {
|
|
|
|
expected = (std::max)(expected, in(l, k, i, j));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(out(i, j), expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <int DataLayout>
|
|
|
|
static void test_innermost_first_dims() {
|
|
|
|
Tensor<float, 4, DataLayout> in(72, 53, 97, 113);
|
|
|
|
Tensor<float, 2, DataLayout> out(72, 53);
|
|
|
|
in.setRandom();
|
|
|
|
|
|
|
|
// Reduce on the innermost dimensions.
|
2015-02-06 18:51:59 +08:00
|
|
|
#ifndef EIGEN_HAS_CONSTEXPR
|
2015-01-15 07:46:04 +08:00
|
|
|
array<int, 2> reduction_axis;
|
|
|
|
reduction_axis[0] = 2;
|
|
|
|
reduction_axis[1] = 3;
|
|
|
|
#else
|
|
|
|
// This triggers the use of packets for RowMajor.
|
|
|
|
Eigen::IndexList<Eigen::type2index<2>, Eigen::type2index<3>> reduction_axis;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
out = in.maximum(reduction_axis);
|
|
|
|
|
|
|
|
for (int i = 0; i < 72; ++i) {
|
|
|
|
for (int j = 0; j < 53; ++j) {
|
|
|
|
float expected = -1e10f;
|
|
|
|
for (int k = 0; k < 97; ++k) {
|
|
|
|
for (int l = 0; l < 113; ++l) {
|
|
|
|
expected = (std::max)(expected, in(i, j, k, l));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(out(i, j), expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-10-10 06:36:23 +08:00
|
|
|
|
2015-01-29 02:02:47 +08:00
|
|
|
template <int DataLayout>
|
|
|
|
static void test_reduce_middle_dims() {
|
|
|
|
Tensor<float, 4, DataLayout> in(72, 53, 97, 113);
|
|
|
|
Tensor<float, 2, DataLayout> out(72, 53);
|
|
|
|
in.setRandom();
|
|
|
|
|
|
|
|
// Reduce on the innermost dimensions.
|
2015-02-06 18:51:59 +08:00
|
|
|
#ifndef EIGEN_HAS_CONSTEXPR
|
2015-01-29 02:02:47 +08:00
|
|
|
array<int, 2> reduction_axis;
|
|
|
|
reduction_axis[0] = 1;
|
|
|
|
reduction_axis[1] = 2;
|
|
|
|
#else
|
|
|
|
// This triggers the use of packets for RowMajor.
|
|
|
|
Eigen::IndexList<Eigen::type2index<1>, Eigen::type2index<2>> reduction_axis;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
out = in.maximum(reduction_axis);
|
|
|
|
|
|
|
|
for (int i = 0; i < 72; ++i) {
|
|
|
|
for (int j = 0; j < 113; ++j) {
|
|
|
|
float expected = -1e10f;
|
|
|
|
for (int k = 0; k < 53; ++k) {
|
|
|
|
for (int l = 0; l < 97; ++l) {
|
|
|
|
expected = (std::max)(expected, in(i, k, l, j));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VERIFY_IS_APPROX(out(i, j), expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 07:46:04 +08:00
|
|
|
void test_cxx11_tensor_reduction() {
|
|
|
|
CALL_SUBTEST(test_simple_reductions<ColMajor>());
|
|
|
|
CALL_SUBTEST(test_simple_reductions<RowMajor>());
|
|
|
|
CALL_SUBTEST(test_full_reductions<ColMajor>());
|
|
|
|
CALL_SUBTEST(test_full_reductions<RowMajor>());
|
|
|
|
CALL_SUBTEST(test_user_defined_reductions<ColMajor>());
|
|
|
|
CALL_SUBTEST(test_user_defined_reductions<RowMajor>());
|
|
|
|
CALL_SUBTEST(test_tensor_maps<ColMajor>());
|
|
|
|
CALL_SUBTEST(test_tensor_maps<RowMajor>());
|
|
|
|
CALL_SUBTEST(test_static_dims<ColMajor>());
|
|
|
|
CALL_SUBTEST(test_static_dims<RowMajor>());
|
|
|
|
CALL_SUBTEST(test_innermost_last_dims<ColMajor>());
|
2015-01-29 02:02:47 +08:00
|
|
|
CALL_SUBTEST(test_innermost_last_dims<RowMajor>());
|
2015-01-15 07:46:04 +08:00
|
|
|
CALL_SUBTEST(test_innermost_first_dims<ColMajor>());
|
2015-01-29 02:02:47 +08:00
|
|
|
CALL_SUBTEST(test_innermost_first_dims<RowMajor>());
|
|
|
|
CALL_SUBTEST(test_reduce_middle_dims<ColMajor>());
|
|
|
|
CALL_SUBTEST(test_reduce_middle_dims<RowMajor>());
|
2014-10-02 11:38:22 +08:00
|
|
|
}
|