Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
Crayon鑫
Paddle
提交
9a14ca91
P
Paddle
项目概览
Crayon鑫
/
Paddle
与 Fork 源项目一致
Fork自
PaddlePaddle / Paddle
通知
1
Star
1
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
1
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
P
Paddle
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
1
Issue
1
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
9a14ca91
编写于
10月 18, 2018
作者:
J
jerrywgz
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
test=develop
上级
4c9884e7
变更
5
显示空白变更内容
内联
并排
Showing
5 changed file
with
118 addition
and
136 deletion
+118
-136
paddle/fluid/API.spec
paddle/fluid/API.spec
+1
-1
paddle/fluid/operators/roi_align_op.cc
paddle/fluid/operators/roi_align_op.cc
+1
-2
paddle/fluid/operators/roi_align_op.cu
paddle/fluid/operators/roi_align_op.cu
+47
-54
paddle/fluid/operators/roi_align_op.h
paddle/fluid/operators/roi_align_op.h
+67
-78
python/paddle/fluid/layers/nn.py
python/paddle/fluid/layers/nn.py
+2
-1
未找到文件。
paddle/fluid/API.spec
浏览文件 @
9a14ca91
...
@@ -114,7 +114,7 @@ paddle.fluid.layers.pad ArgSpec(args=['x', 'paddings', 'pad_value', 'name'], var
...
@@ -114,7 +114,7 @@ paddle.fluid.layers.pad ArgSpec(args=['x', 'paddings', 'pad_value', 'name'], var
paddle.fluid.layers.pad_constant_like ArgSpec(args=['x', 'y', 'pad_value', 'name'], varargs=None, keywords=None, defaults=(0.0, None))
paddle.fluid.layers.pad_constant_like ArgSpec(args=['x', 'y', 'pad_value', 'name'], varargs=None, keywords=None, defaults=(0.0, None))
paddle.fluid.layers.label_smooth ArgSpec(args=['label', 'prior_dist', 'epsilon', 'dtype', 'name'], varargs=None, keywords=None, defaults=(None, 0.1, 'float32', None))
paddle.fluid.layers.label_smooth ArgSpec(args=['label', 'prior_dist', 'epsilon', 'dtype', 'name'], varargs=None, keywords=None, defaults=(None, 0.1, 'float32', None))
paddle.fluid.layers.roi_pool ArgSpec(args=['input', 'rois', 'pooled_height', 'pooled_width', 'spatial_scale'], varargs=None, keywords=None, defaults=(1, 1, 1.0))
paddle.fluid.layers.roi_pool ArgSpec(args=['input', 'rois', 'pooled_height', 'pooled_width', 'spatial_scale'], varargs=None, keywords=None, defaults=(1, 1, 1.0))
paddle.fluid.layers.roi_align ArgSpec(args=['input', 'rois', 'pooled_height', 'pooled_width', 'spatial_scale', 'sampling_ratio'
], varargs=None, keywords=None, defaults=(1, 1, 1.0, -1
))
paddle.fluid.layers.roi_align ArgSpec(args=['input', 'rois', 'pooled_height', 'pooled_width', 'spatial_scale', 'sampling_ratio'
, 'name'], varargs=None, keywords=None, defaults=(1, 1, 1.0, -1, None
))
paddle.fluid.layers.dice_loss ArgSpec(args=['input', 'label', 'epsilon'], varargs=None, keywords=None, defaults=(1e-05,))
paddle.fluid.layers.dice_loss ArgSpec(args=['input', 'label', 'epsilon'], varargs=None, keywords=None, defaults=(1e-05,))
paddle.fluid.layers.image_resize ArgSpec(args=['input', 'out_shape', 'scale', 'name', 'resample'], varargs=None, keywords=None, defaults=(None, None, None, 'BILINEAR'))
paddle.fluid.layers.image_resize ArgSpec(args=['input', 'out_shape', 'scale', 'name', 'resample'], varargs=None, keywords=None, defaults=(None, None, None, 'BILINEAR'))
paddle.fluid.layers.image_resize_short ArgSpec(args=['input', 'out_short_len', 'resample'], varargs=None, keywords=None, defaults=('BILINEAR',))
paddle.fluid.layers.image_resize_short ArgSpec(args=['input', 'out_short_len', 'resample'], varargs=None, keywords=None, defaults=('BILINEAR',))
...
...
paddle/fluid/operators/roi_align_op.cc
浏览文件 @
9a14ca91
...
@@ -94,7 +94,7 @@ class ROIAlignOpMaker : public framework::OpProtoAndCheckerMaker {
...
@@ -94,7 +94,7 @@ class ROIAlignOpMaker : public framework::OpProtoAndCheckerMaker {
void
Make
()
override
{
void
Make
()
override
{
AddInput
(
"X"
,
AddInput
(
"X"
,
"(Tensor), "
"(Tensor), "
"
t
he input of ROIAlignOp. "
"
T
he input of ROIAlignOp. "
"The format of input tensor is NCHW. Where N is batch size, "
"The format of input tensor is NCHW. Where N is batch size, "
"C is the number of input channels, "
"C is the number of input channels, "
"H is the height of the feature, and "
"H is the height of the feature, and "
...
@@ -104,7 +104,6 @@ class ROIAlignOpMaker : public framework::OpProtoAndCheckerMaker {
...
@@ -104,7 +104,6 @@ class ROIAlignOpMaker : public framework::OpProtoAndCheckerMaker {
"ROIs (Regions of Interest) to pool over. "
"ROIs (Regions of Interest) to pool over. "
"should be a 2-D LoDTensor of shape (num_rois, 4)"
"should be a 2-D LoDTensor of shape (num_rois, 4)"
"given as [[x1, y1, x2, y2], …]. "
"given as [[x1, y1, x2, y2], …]. "
"Where batch_id is the id of the data, "
"(x1, y1) is the top left coordinates, and "
"(x1, y1) is the top left coordinates, and "
"(x2, y2) is the bottom right coordinates."
);
"(x2, y2) is the bottom right coordinates."
);
AddOutput
(
"Out"
,
AddOutput
(
"Out"
,
...
...
paddle/fluid/operators/roi_align_op.cu
浏览文件 @
9a14ca91
...
@@ -34,17 +34,13 @@ static inline int NumBlocks(const int N) {
...
@@ -34,17 +34,13 @@ static inline int NumBlocks(const int N) {
i += blockDim.x * gridDim.x)
i += blockDim.x * gridDim.x)
template
<
class
T
>
template
<
class
T
>
__device__
T
bilinear_i
nterpolate
(
const
T
*
input_data
,
const
int
height
,
__device__
T
BilinearI
nterpolate
(
const
T
*
input_data
,
const
int
height
,
const
int
width
,
T
y
,
T
x
)
{
const
int
width
,
T
y
,
T
x
)
{
if
(
y
<
-
1.0
||
y
>
height
||
x
<
-
1.0
||
x
>
width
)
{
if
(
y
<
-
1.0
||
y
>
height
||
x
<
-
1.0
||
x
>
width
)
{
return
0
;
return
0
;
}
}
if
(
y
<=
0
)
{
y
=
y
<=
0
?
0
:
y
;
y
=
0
;
x
=
x
<=
0
?
0
:
x
;
}
if
(
x
<=
0
)
{
x
=
0
;
}
int
y_low
=
static_cast
<
int
>
(
y
);
int
y_low
=
static_cast
<
int
>
(
y
);
int
x_low
=
static_cast
<
int
>
(
x
);
int
x_low
=
static_cast
<
int
>
(
x
);
int
y_high
;
int
y_high
;
...
@@ -75,7 +71,7 @@ __device__ T bilinear_interpolate(const T* input_data, const int height,
...
@@ -75,7 +71,7 @@ __device__ T bilinear_interpolate(const T* input_data, const int height,
}
}
template
<
class
T
>
template
<
class
T
>
__device__
void
bilinear_interpolate_g
radient
(
const
int
height
,
const
int
width
,
__device__
void
BilinearInterpolateG
radient
(
const
int
height
,
const
int
width
,
T
y
,
T
x
,
T
*
w1
,
T
*
w2
,
T
*
w3
,
T
y
,
T
x
,
T
*
w1
,
T
*
w2
,
T
*
w3
,
T
*
w4
,
int
*
x_low
,
int
*
x_high
,
T
*
w4
,
int
*
x_low
,
int
*
x_high
,
int
*
y_low
,
int
*
y_high
)
{
int
*
y_low
,
int
*
y_high
)
{
...
@@ -83,12 +79,8 @@ __device__ void bilinear_interpolate_gradient(const int height, const int width,
...
@@ -83,12 +79,8 @@ __device__ void bilinear_interpolate_gradient(const int height, const int width,
return
;
return
;
}
}
if
(
y
<=
0
)
{
y
=
y
<=
0
?
0
:
y
;
y
=
0
;
x
=
x
<=
0
?
0
:
x
;
}
if
(
x
<=
0
)
{
x
=
0
;
}
*
y_low
=
static_cast
<
int
>
(
y
);
*
y_low
=
static_cast
<
int
>
(
y
);
*
x_low
=
static_cast
<
int
>
(
x
);
*
x_low
=
static_cast
<
int
>
(
x
);
if
(
*
y_low
>=
height
-
1
)
{
if
(
*
y_low
>=
height
-
1
)
{
...
@@ -153,7 +145,7 @@ __global__ void GPUROIAlignForward(
...
@@ -153,7 +145,7 @@ __global__ void GPUROIAlignForward(
const
T
x
=
roi_xmin
+
pw
*
bin_size_w
+
const
T
x
=
roi_xmin
+
pw
*
bin_size_w
+
static_cast
<
T
>
(
ix
+
.5
f
)
*
bin_size_w
/
static_cast
<
T
>
(
ix
+
.5
f
)
*
bin_size_w
/
static_cast
<
T
>
(
roi_bin_grid_w
);
static_cast
<
T
>
(
roi_bin_grid_w
);
T
val
=
bilinear_i
nterpolate
(
offset_input_data
,
height
,
width
,
y
,
x
);
T
val
=
BilinearI
nterpolate
(
offset_input_data
,
height
,
width
,
y
,
x
);
output_val
+=
val
;
output_val
+=
val
;
}
}
}
}
...
@@ -213,7 +205,7 @@ __global__ void GPUROIAlignBackward(const int nthreads, const T* input_rois,
...
@@ -213,7 +205,7 @@ __global__ void GPUROIAlignBackward(const int nthreads, const T* input_rois,
static_cast
<
T
>
(
roi_bin_grid_w
);
static_cast
<
T
>
(
roi_bin_grid_w
);
T
w1
=
0
,
w2
=
0
,
w3
=
0
,
w4
=
0
;
T
w1
=
0
,
w2
=
0
,
w3
=
0
,
w4
=
0
;
int
x_low
=
-
1
,
x_high
=
-
1
,
y_low
=
-
1
,
y_high
=
-
1
;
int
x_low
=
-
1
,
x_high
=
-
1
,
y_low
=
-
1
,
y_high
=
-
1
;
bilinear_interpolate_g
radient
(
height
,
width
,
y
,
x
,
&
w1
,
&
w2
,
&
w3
,
&
w4
,
BilinearInterpolateG
radient
(
height
,
width
,
y
,
x
,
&
w1
,
&
w2
,
&
w3
,
&
w4
,
&
x_low
,
&
x_high
,
&
y_low
,
&
y_high
);
&
x_low
,
&
x_high
,
&
y_low
,
&
y_high
);
T
diff1
=
out_grad_this_bin
*
w1
/
count
;
T
diff1
=
out_grad_this_bin
*
w1
/
count
;
T
diff2
=
out_grad_this_bin
*
w2
/
count
;
T
diff2
=
out_grad_this_bin
*
w2
/
count
;
...
@@ -279,8 +271,8 @@ class GPUROIAlignOpKernel : public framework::OpKernel<T> {
...
@@ -279,8 +271,8 @@ class GPUROIAlignOpKernel : public framework::OpKernel<T> {
}
}
}
}
Tensor
roi_batch_id_list_gpu
;
Tensor
roi_batch_id_list_gpu
;
framework
::
TensorCopy
(
roi_batch_id_list
,
ctx
.
GetPlace
(),
framework
::
TensorCopy
Sync
(
roi_batch_id_list
,
ctx
.
GetPlace
(),
ctx
.
device_context
(),
&
roi_batch_id_list_gpu
);
&
roi_batch_id_list_gpu
);
GPUROIAlignForward
<
GPUROIAlignForward
<
T
><<<
blocks
,
threads
,
0
,
ctx
.
cuda_device_context
().
stream
()
>>>
(
T
><<<
blocks
,
threads
,
0
,
ctx
.
cuda_device_context
().
stream
()
>>>
(
output_size
,
in
->
data
<
T
>
(),
rois
->
data
<
T
>
(),
spatial_scale
,
channels
,
output_size
,
in
->
data
<
T
>
(),
rois
->
data
<
T
>
(),
spatial_scale
,
channels
,
...
@@ -310,7 +302,9 @@ class GPUROIAlignGradOpKernel : public framework::OpKernel<T> {
...
@@ -310,7 +302,9 @@ class GPUROIAlignGradOpKernel : public framework::OpKernel<T> {
int
height
=
in
->
dims
()[
2
];
int
height
=
in
->
dims
()[
2
];
int
width
=
in
->
dims
()[
3
];
int
width
=
in
->
dims
()[
3
];
if
(
in_grad
)
{
if
(
!
in_grad
)
{
return
;
}
Tensor
roi_batch_id_list
;
Tensor
roi_batch_id_list
;
roi_batch_id_list
.
Resize
({
rois_num
});
roi_batch_id_list
.
Resize
({
rois_num
});
int
*
roi_batch_id_data
=
int
*
roi_batch_id_data
=
...
@@ -323,8 +317,8 @@ class GPUROIAlignGradOpKernel : public framework::OpKernel<T> {
...
@@ -323,8 +317,8 @@ class GPUROIAlignGradOpKernel : public framework::OpKernel<T> {
}
}
}
}
Tensor
roi_batch_id_list_gpu
;
Tensor
roi_batch_id_list_gpu
;
framework
::
TensorCopy
(
roi_batch_id_list
,
ctx
.
GetPlace
(),
framework
::
TensorCopySync
(
roi_batch_id_list
,
ctx
.
GetPlace
(),
ctx
.
device_context
(),
&
roi_batch_id_list_gpu
);
&
roi_batch_id_list_gpu
);
in_grad
->
mutable_data
<
T
>
(
ctx
.
GetPlace
());
in_grad
->
mutable_data
<
T
>
(
ctx
.
GetPlace
());
math
::
SetConstant
<
Place
,
T
>
set_zero
;
math
::
SetConstant
<
Place
,
T
>
set_zero
;
...
@@ -343,7 +337,6 @@ class GPUROIAlignGradOpKernel : public framework::OpKernel<T> {
...
@@ -343,7 +337,6 @@ class GPUROIAlignGradOpKernel : public framework::OpKernel<T> {
in_grad
->
mutable_data
<
T
>
(
ctx
.
GetPlace
()));
in_grad
->
mutable_data
<
T
>
(
ctx
.
GetPlace
()));
}
}
}
}
}
};
};
}
// namespace operators
}
// namespace operators
...
...
paddle/fluid/operators/roi_align_op.h
浏览文件 @
9a14ca91
...
@@ -24,7 +24,7 @@ using LoDTensor = framework::LoDTensor;
...
@@ -24,7 +24,7 @@ using LoDTensor = framework::LoDTensor;
static
constexpr
int
kROISize
=
4
;
static
constexpr
int
kROISize
=
4
;
template
<
class
T
>
template
<
class
T
>
void
pre_calc_for_bilinear_i
nterpolate
(
void
PreCalcForBilinearI
nterpolate
(
const
platform
::
DeviceContext
&
ctx
,
const
int
height
,
const
int
width
,
const
platform
::
DeviceContext
&
ctx
,
const
int
height
,
const
int
width
,
const
int
pooled_height
,
const
int
pooled_width
,
const
int
iy_upper
,
const
int
pooled_height
,
const
int
pooled_width
,
const
int
iy_upper
,
const
int
ix_upper
,
T
roi_ymin
,
T
roi_xmin
,
T
bin_size_h
,
T
bin_size_w
,
const
int
ix_upper
,
T
roi_ymin
,
T
roi_xmin
,
T
bin_size_h
,
T
bin_size_w
,
...
@@ -53,12 +53,8 @@ void pre_calc_for_bilinear_interpolate(
...
@@ -53,12 +53,8 @@ void pre_calc_for_bilinear_interpolate(
pre_calc_index
+=
1
;
pre_calc_index
+=
1
;
continue
;
continue
;
}
}
if
(
y
<=
0
)
{
y
=
y
<=
0
?
0
:
y
;
y
=
0
;
x
=
x
<=
0
?
0
:
x
;
}
if
(
x
<=
0
)
{
x
=
0
;
}
int
y_low
=
static_cast
<
int
>
(
y
);
int
y_low
=
static_cast
<
int
>
(
y
);
int
x_low
=
static_cast
<
int
>
(
x
);
int
x_low
=
static_cast
<
int
>
(
x
);
...
@@ -104,12 +100,8 @@ void bilinear_interpolate_gradient(const int height, const int width, T y, T x,
...
@@ -104,12 +100,8 @@ void bilinear_interpolate_gradient(const int height, const int width, T y, T x,
x_low
=
x_high
=
y_low
=
y_high
=
-
1
;
x_low
=
x_high
=
y_low
=
y_high
=
-
1
;
return
;
return
;
}
}
if
(
y
<=
0
)
{
y
=
y
<=
0
?
0
:
y
;
y
=
0
;
x
=
x
<=
0
?
0
:
x
;
}
if
(
x
<=
0
)
{
x
=
0
;
}
y_low
=
static_cast
<
int
>
(
y
);
y_low
=
static_cast
<
int
>
(
y
);
x_low
=
static_cast
<
int
>
(
x
);
x_low
=
static_cast
<
int
>
(
x
);
if
(
y_low
>=
height
-
1
)
{
if
(
y_low
>=
height
-
1
)
{
...
@@ -139,7 +131,6 @@ void bilinear_interpolate_gradient(const int height, const int width, T y, T x,
...
@@ -139,7 +131,6 @@ void bilinear_interpolate_gradient(const int height, const int width, T y, T x,
*
(
batch_grad_data
+
y_high
*
width
+
x_low
)
+=
diff3
;
*
(
batch_grad_data
+
y_high
*
width
+
x_low
)
+=
diff3
;
*
(
batch_grad_data
+
y_high
*
width
+
x_high
)
+=
diff4
;
*
(
batch_grad_data
+
y_high
*
width
+
x_high
)
+=
diff4
;
}
}
return
;
}
}
template
<
typename
DeviceContext
,
typename
T
>
template
<
typename
DeviceContext
,
typename
T
>
...
@@ -214,7 +205,7 @@ class CPUROIAlignOpKernel : public framework::OpKernel<T> {
...
@@ -214,7 +205,7 @@ class CPUROIAlignOpKernel : public framework::OpKernel<T> {
pre_pos
.
Resize
({
pre_size
,
kROISize
});
pre_pos
.
Resize
({
pre_size
,
kROISize
});
pre_w
.
Resize
({
pre_size
,
kROISize
});
pre_w
.
Resize
({
pre_size
,
kROISize
});
pre_calc_for_bilinear_i
nterpolate
(
PreCalcForBilinearI
nterpolate
(
dev_ctx
,
height
,
width
,
pooled_height
,
pooled_width
,
roi_bin_grid_h
,
dev_ctx
,
height
,
width
,
pooled_height
,
pooled_width
,
roi_bin_grid_h
,
roi_bin_grid_w
,
roi_ymin
,
roi_xmin
,
bin_size_h
,
bin_size_w
,
roi_bin_grid_w
,
roi_ymin
,
roi_xmin
,
bin_size_h
,
bin_size_w
,
roi_bin_grid_h
,
roi_bin_grid_w
,
&
pre_pos
,
&
pre_w
);
roi_bin_grid_h
,
roi_bin_grid_w
,
&
pre_pos
,
&
pre_w
);
...
@@ -245,7 +236,6 @@ class CPUROIAlignOpKernel : public framework::OpKernel<T> {
...
@@ -245,7 +236,6 @@ class CPUROIAlignOpKernel : public framework::OpKernel<T> {
}
}
rois_data
+=
roi_stride
[
0
];
rois_data
+=
roi_stride
[
0
];
}
}
return
;
}
}
};
};
...
@@ -264,7 +254,9 @@ class CPUROIAlignGradOpKernel : public framework::OpKernel<T> {
...
@@ -264,7 +254,9 @@ class CPUROIAlignGradOpKernel : public framework::OpKernel<T> {
auto
spatial_scale
=
ctx
.
Attr
<
float
>
(
"spatial_scale"
);
auto
spatial_scale
=
ctx
.
Attr
<
float
>
(
"spatial_scale"
);
auto
sampling_ratio
=
ctx
.
Attr
<
int
>
(
"sampling_ratio"
);
auto
sampling_ratio
=
ctx
.
Attr
<
int
>
(
"sampling_ratio"
);
auto
in_dims
=
in
->
dims
();
auto
in_dims
=
in
->
dims
();
if
(
in_grad
)
{
if
(
!
in_grad
)
{
return
;
}
int
channels
=
in_dims
[
1
];
int
channels
=
in_dims
[
1
];
int
height
=
in_dims
[
2
];
int
height
=
in_dims
[
2
];
int
width
=
in_dims
[
3
];
int
width
=
in_dims
[
3
];
...
@@ -298,8 +290,7 @@ class CPUROIAlignGradOpKernel : public framework::OpKernel<T> {
...
@@ -298,8 +290,7 @@ class CPUROIAlignGradOpKernel : public framework::OpKernel<T> {
T
roi_ymax
=
rois_data
[
3
]
*
spatial_scale
;
T
roi_ymax
=
rois_data
[
3
]
*
spatial_scale
;
T
roi_width
=
std
::
max
(
roi_xmax
-
roi_xmin
,
static_cast
<
T
>
(
1.
));
T
roi_width
=
std
::
max
(
roi_xmax
-
roi_xmin
,
static_cast
<
T
>
(
1.
));
T
roi_height
=
std
::
max
(
roi_ymax
-
roi_ymin
,
static_cast
<
T
>
(
1.
));
T
roi_height
=
std
::
max
(
roi_ymax
-
roi_ymin
,
static_cast
<
T
>
(
1.
));
T
bin_size_h
=
T
bin_size_h
=
static_cast
<
T
>
(
roi_height
)
/
static_cast
<
T
>
(
pooled_height
);
static_cast
<
T
>
(
roi_height
)
/
static_cast
<
T
>
(
pooled_height
);
T
bin_size_w
=
static_cast
<
T
>
(
roi_width
)
/
static_cast
<
T
>
(
pooled_width
);
T
bin_size_w
=
static_cast
<
T
>
(
roi_width
)
/
static_cast
<
T
>
(
pooled_width
);
for
(
int
c
=
0
;
c
<
channels
;
++
c
)
{
for
(
int
c
=
0
;
c
<
channels
;
++
c
)
{
T
*
batch_grad_data
=
T
*
batch_grad_data
=
...
@@ -336,8 +327,6 @@ class CPUROIAlignGradOpKernel : public framework::OpKernel<T> {
...
@@ -336,8 +327,6 @@ class CPUROIAlignGradOpKernel : public framework::OpKernel<T> {
rois_data
+=
roi_stride
[
0
];
rois_data
+=
roi_stride
[
0
];
}
}
}
}
return
;
}
};
};
}
// namespace operators
}
// namespace operators
}
// namespace paddle
}
// namespace paddle
python/paddle/fluid/layers/nn.py
浏览文件 @
9a14ca91
...
@@ -5184,7 +5184,8 @@ def roi_align(input,
...
@@ -5184,7 +5184,8 @@ def roi_align(input,
pooled_height
=
1
,
pooled_height
=
1
,
pooled_width
=
1
,
pooled_width
=
1
,
spatial_scale
=
1.0
,
spatial_scale
=
1.0
,
sampling_ratio
=-
1
):
sampling_ratio
=-
1
,
name
=
None
):
"""
"""
${comment}
${comment}
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录