Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
PaddlePaddle
Paddle
提交
3b4bdc08
P
Paddle
项目概览
PaddlePaddle
/
Paddle
大约 2 年 前同步成功
通知
2325
Star
20933
Fork
5424
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
1423
列表
看板
标记
里程碑
合并请求
543
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
P
Paddle
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
1,423
Issue
1,423
列表
看板
标记
里程碑
合并请求
543
合并请求
543
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
未验证
提交
3b4bdc08
编写于
6月 13, 2023
作者:
石
石晓伟
提交者:
GitHub
6月 13, 2023
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
add the span data structure (#54549)
上级
10188e8f
变更
3
显示空白变更内容
内联
并排
Showing
3 changed file
with
1063 addition
and
0 deletion
+1063
-0
paddle/utils/CMakeLists.txt
paddle/utils/CMakeLists.txt
+4
-0
paddle/utils/span.h
paddle/utils/span.h
+452
-0
paddle/utils/span_test.cc
paddle/utils/span_test.cc
+607
-0
未找到文件。
paddle/utils/CMakeLists.txt
浏览文件 @
3b4bdc08
...
...
@@ -12,6 +12,10 @@ cc_test(
variant_test
SRCS variant_test.cc
DEPS gtest
)
cc_test
(
span_test
SRCS span_test.cc
DEPS gtest
)
if
(
NOT
((
NOT WITH_PYTHON
)
AND ON_INFER
))
cc_library
(
...
...
paddle/utils/span.h
0 → 100644
浏览文件 @
3b4bdc08
// Copyright (c) 2023 PaddlePaddle Authors. All Rights Reserved.
// This file copy from https://github.com/tcbrindle/span
// Modified the following points
// 1. remove macros for backward compatibility with pre-C++17 standards
// 2. instantiated namespace name with paddle
/*
This is an implementation of C++20's std::span
http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2019/n4820.pdf
*/
// Copyright Tristan Brindle 2018.
// Distributed under the Boost Software License, Version 1.0.
// (See accompanying file ../../LICENSE_1_0.txt or copy at
// https://www.boost.org/LICENSE_1_0.txt)
#pragma once
#include <array>
#include <cstddef>
#include <cstdint>
#include <exception>
#include <limits>
#include <type_traits>
#ifdef SPAN_THROW_ON_CONTRACT_VIOLATION
#include <cstdio>
#include <stdexcept>
#endif
#if __cplusplus >= 201703L || (defined(_MSVC_LANG) && _MSVC_LANG >= 201703L)
#define NO_DISCARD [[nodiscard]]
#else
#define NO_DISCARD
#endif
namespace
paddle
{
#if defined(SPAN_THROW_ON_CONTRACT_VIOLATION)
struct
contract_violation_error
:
std
::
logic_error
{
explicit
contract_violation_error
(
const
char
*
msg
)
:
std
::
logic_error
(
msg
)
{}
};
inline
void
contract_violation
(
const
char
*
msg
)
{
throw
contract_violation_error
(
msg
);
}
#else
[[
noreturn
]]
void
contract_violation
(
const
char
*
/*unused*/
)
{
std
::
terminate
();
}
#endif
#if !defined(SPAN_NO_CONTRACT_CHECKING)
#define SPAN_STRINGIFY(cond) #cond
#define SPAN_EXPECT(cond) \
cond ? (void)0 : contract_violation("Expected " SPAN_STRINGIFY(cond))
#else
#define SPAN_EXPECT(cond)
#endif
#ifdef __cpp_inline_variables
inline
constexpr
std
::
size_t
dynamic_extent
=
std
::
numeric_limits
<
std
::
size_t
>::
max
();
#else
constexpr
std
::
size_t
dynamic_extent
=
std
::
numeric_limits
<
std
::
size_t
>::
max
();
#endif
template
<
typename
ElementType
,
std
::
size_t
Extent
=
dynamic_extent
>
class
span
;
namespace
detail
{
#ifdef __cpp_lib_byte
using
byte
=
std
::
byte
;
#else
using
byte
=
unsigned
char
;
#endif
#ifdef __cpp_lib_nonmember_container_access
using
std
::
data
;
using
std
::
size
;
#else
template
<
class
C
>
constexpr
auto
size
(
const
C
&
c
)
->
decltype
(
c
.
size
())
{
return
c
.
size
();
}
template
<
class
T
,
std
::
size_t
N
>
constexpr
std
::
size_t
size
(
const
T
(
&
)[
N
])
noexcept
{
return
N
;
}
template
<
class
C
>
constexpr
auto
data
(
C
&
c
)
->
decltype
(
c
.
data
())
{
// NOLINT
return
c
.
data
();
}
template
<
class
C
>
constexpr
auto
data
(
const
C
&
c
)
->
decltype
(
c
.
data
())
{
return
c
.
data
();
}
template
<
class
T
,
std
::
size_t
N
>
constexpr
T
*
data
(
T
(
&
array
)[
N
])
noexcept
{
return
array
;
}
template
<
class
E
>
constexpr
const
E
*
data
(
std
::
initializer_list
<
E
>
il
)
noexcept
{
return
il
.
begin
();
}
#endif
#ifdef __cpp_lib_void_t
using
std
::
void_t
;
#else
template
<
typename
...
>
using
void_t
=
void
;
#endif
template
<
typename
E
,
std
::
size_t
S
>
struct
span_storage
{
constexpr
span_storage
()
noexcept
=
default
;
constexpr
span_storage
(
E
*
ptr
,
std
::
size_t
/*unused*/
)
noexcept
:
ptr
{
ptr
}
{}
E
*
ptr
{};
static
constexpr
std
::
size_t
size
{
S
};
};
template
<
typename
E
>
struct
span_storage
<
E
,
dynamic_extent
>
{
constexpr
span_storage
()
noexcept
=
default
;
constexpr
span_storage
(
E
*
ptr
,
std
::
size_t
size
)
noexcept
:
ptr
{
ptr
},
size
{
size
}
{}
E
*
ptr
{};
std
::
size_t
size
{};
};
template
<
typename
>
struct
is_span
:
std
::
false_type
{};
template
<
typename
T
,
std
::
size_t
S
>
struct
is_span
<
span
<
T
,
S
>>
:
std
::
true_type
{};
template
<
typename
>
struct
is_std_array
:
std
::
false_type
{};
template
<
typename
T
,
std
::
size_t
N
>
struct
is_std_array
<
std
::
array
<
T
,
N
>>
:
std
::
true_type
{};
template
<
typename
,
typename
=
void
>
struct
has_size_and_data
:
std
::
false_type
{};
template
<
typename
T
>
struct
has_size_and_data
<
T
,
detail
::
void_t
<
decltype
(
detail
::
size
(
std
::
declval
<
T
>
())),
decltype
(
detail
::
data
(
std
::
declval
<
T
>
()))
>>
:
std
::
true_type
{};
template
<
typename
C
,
typename
U
=
typename
std
::
remove_cv
<
typename
std
::
remove_reference
<
C
>
::
type
>::
type
>
struct
is_container
{
static
constexpr
bool
value
=
!
is_span
<
U
>::
value
&&
!
is_std_array
<
U
>::
value
&&
!
std
::
is_array
<
U
>::
value
&&
has_size_and_data
<
C
>::
value
;
};
template
<
typename
,
typename
,
typename
=
void
>
struct
is_container_element_type_compatible
:
std
::
false_type
{};
template
<
typename
T
,
typename
E
>
struct
is_container_element_type_compatible
<
T
,
E
,
typename
std
::
enable_if
<
!
std
::
is_same
<
typename
std
::
remove_cv
<
decltype
(
detail
::
data
(
std
::
declval
<
T
>
()))
>::
type
,
void
>::
value
&&
std
::
is_convertible
<
typename
std
::
remove_pointer
<
decltype
(
detail
::
data
(
std
::
declval
<
T
>
()))
>::
type
(
*
)[],
E
(
*
)[]
>::
value
>::
type
>
:
std
::
true_type
{};
template
<
typename
,
typename
=
std
::
size_t
>
struct
is_complete
:
std
::
false_type
{};
template
<
typename
T
>
struct
is_complete
<
T
,
decltype
(
sizeof
(
T
))
>
:
std
::
true_type
{};
}
// namespace detail
template
<
typename
ElementType
,
std
::
size_t
Extent
>
class
span
{
static_assert
(
std
::
is_object
<
ElementType
>::
value
,
"A span's ElementType must be an object type (not a "
"reference type or void)"
);
static_assert
(
detail
::
is_complete
<
ElementType
>::
value
,
"A span's ElementType must be a complete type (not a forward "
"declaration)"
);
static_assert
(
!
std
::
is_abstract
<
ElementType
>::
value
,
"A span's ElementType cannot be an abstract class type"
);
using
storage_type
=
detail
::
span_storage
<
ElementType
,
Extent
>
;
public:
using
element_type
=
ElementType
;
using
value_type
=
typename
std
::
remove_cv
<
ElementType
>::
type
;
using
size_type
=
std
::
size_t
;
using
difference_type
=
std
::
ptrdiff_t
;
using
pointer
=
element_type
*
;
using
const_pointer
=
const
element_type
*
;
using
reference
=
element_type
&
;
using
const_reference
=
const
element_type
&
;
using
iterator
=
pointer
;
using
reverse_iterator
=
std
::
reverse_iterator
<
iterator
>
;
static
constexpr
size_type
extent
=
Extent
;
// [span.cons], span constructors, copy, assignment, and destructor
template
<
std
::
size_t
E
=
Extent
,
typename
std
::
enable_if
<
E
==
dynamic_extent
||
E
==
0
,
int
>
::
type
=
0
>
constexpr
span
()
noexcept
{}
constexpr
span
(
pointer
ptr
,
size_type
count
)
:
storage_
(
ptr
,
count
)
{
SPAN_EXPECT
(
extent
==
dynamic_extent
||
count
==
extent
);
}
constexpr
span
(
pointer
first_elem
,
pointer
last_elem
)
:
storage_
(
first_elem
,
last_elem
-
first_elem
)
{
SPAN_EXPECT
(
extent
==
dynamic_extent
||
last_elem
-
first_elem
==
static_cast
<
std
::
ptrdiff_t
>
(
extent
));
}
template
<
std
::
size_t
N
,
std
::
size_t
E
=
Extent
,
typename
std
::
enable_if
<
(
E
==
dynamic_extent
||
N
==
E
)
&&
detail
::
is_container_element_type_compatible
<
element_type
(
&
)[
N
],
ElementType
>
::
value
,
int
>::
type
=
0
>
constexpr
span
(
element_type
(
&
arr
)[
N
])
noexcept
// NOLINT
:
storage_
(
arr
,
N
)
{}
template
<
typename
T
,
std
::
size_t
N
,
std
::
size_t
E
=
Extent
,
typename
std
::
enable_if
<
(
E
==
dynamic_extent
||
N
==
E
)
&&
detail
::
is_container_element_type_compatible
<
std
::
array
<
T
,
N
>
&
,
ElementType
>::
value
,
int
>::
type
=
0
>
constexpr
span
(
std
::
array
<
T
,
N
>&
arr
)
noexcept
// NOLINT
:
storage_
(
arr
.
data
(),
N
)
{}
template
<
typename
T
,
std
::
size_t
N
,
std
::
size_t
E
=
Extent
,
typename
std
::
enable_if
<
(
E
==
dynamic_extent
||
N
==
E
)
&&
detail
::
is_container_element_type_compatible
<
const
std
::
array
<
T
,
N
>
&
,
ElementType
>::
value
,
int
>::
type
=
0
>
constexpr
span
(
const
std
::
array
<
T
,
N
>&
arr
)
noexcept
// NOLINT
:
storage_
(
arr
.
data
(),
N
)
{}
template
<
typename
Container
,
std
::
size_t
E
=
Extent
,
typename
std
::
enable_if
<
E
==
dynamic_extent
&&
detail
::
is_container
<
Container
>
::
value
&&
detail
::
is_container_element_type_compatible
<
Container
&
,
ElementType
>::
value
,
int
>::
type
=
0
>
constexpr
span
(
Container
&
cont
)
// NOLINT
:
storage_
(
detail
::
data
(
cont
),
detail
::
size
(
cont
))
{}
template
<
typename
Container
,
std
::
size_t
E
=
Extent
,
typename
std
::
enable_if
<
E
==
dynamic_extent
&&
detail
::
is_container
<
Container
>
::
value
&&
detail
::
is_container_element_type_compatible
<
const
Container
&
,
ElementType
>::
value
,
int
>::
type
=
0
>
constexpr
span
(
const
Container
&
cont
)
// NOLINT
:
storage_
(
detail
::
data
(
cont
),
detail
::
size
(
cont
))
{}
constexpr
span
(
const
span
&
other
)
noexcept
=
default
;
template
<
typename
OtherElementType
,
std
::
size_t
OtherExtent
,
typename
std
::
enable_if
<
(
Extent
==
dynamic_extent
||
OtherExtent
==
dynamic_extent
||
Extent
==
OtherExtent
)
&&
std
::
is_convertible
<
OtherElementType
(
*
)[],
ElementType
(
*
)[]>
::
value
,
int
>::
type
=
0
>
constexpr
span
(
const
span
<
OtherElementType
,
OtherExtent
>&
other
)
noexcept
:
storage_
(
other
.
data
(),
other
.
size
())
{}
~
span
()
noexcept
=
default
;
constexpr
span
&
operator
=
(
const
span
&
other
)
noexcept
=
default
;
// [span.sub], span subviews
template
<
std
::
size_t
Count
>
constexpr
span
<
element_type
,
Count
>
first
()
const
{
SPAN_EXPECT
(
Count
<=
size
());
return
{
data
(),
Count
};
}
template
<
std
::
size_t
Count
>
constexpr
span
<
element_type
,
Count
>
last
()
const
{
SPAN_EXPECT
(
Count
<=
size
());
return
{
data
()
+
(
size
()
-
Count
),
Count
};
}
template
<
std
::
size_t
Offset
,
std
::
size_t
Count
=
dynamic_extent
>
using
subspan_return_t
=
span
<
ElementType
,
Count
!=
dynamic_extent
?
Count
:
(
Extent
!=
dynamic_extent
?
Extent
-
Offset
:
dynamic_extent
)
>
;
template
<
std
::
size_t
Offset
,
std
::
size_t
Count
=
dynamic_extent
>
constexpr
subspan_return_t
<
Offset
,
Count
>
subspan
()
const
{
SPAN_EXPECT
(
Offset
<=
size
()
&&
(
Count
==
dynamic_extent
||
Offset
+
Count
<=
size
()));
return
{
data
()
+
Offset
,
Count
!=
dynamic_extent
?
Count
:
size
()
-
Offset
};
}
constexpr
span
<
element_type
,
dynamic_extent
>
first
(
size_type
count
)
const
{
SPAN_EXPECT
(
count
<=
size
());
return
{
data
(),
count
};
}
constexpr
span
<
element_type
,
dynamic_extent
>
last
(
size_type
count
)
const
{
SPAN_EXPECT
(
count
<=
size
());
return
{
data
()
+
(
size
()
-
count
),
count
};
}
constexpr
span
<
element_type
,
dynamic_extent
>
subspan
(
size_type
offset
,
size_type
count
=
dynamic_extent
)
const
{
SPAN_EXPECT
(
offset
<=
size
()
&&
(
count
==
dynamic_extent
||
offset
+
count
<=
size
()));
return
{
data
()
+
offset
,
count
==
dynamic_extent
?
size
()
-
offset
:
count
};
}
// [span.obs], span observers
constexpr
size_type
size
()
const
noexcept
{
return
storage_
.
size
;
}
constexpr
size_type
size_bytes
()
const
noexcept
{
return
size
()
*
sizeof
(
element_type
);
}
NO_DISCARD
constexpr
bool
empty
()
const
noexcept
{
return
size
()
==
0
;
}
// [span.elem], span element access
constexpr
reference
operator
[](
size_type
idx
)
const
{
SPAN_EXPECT
(
idx
<
size
());
return
*
(
data
()
+
idx
);
}
constexpr
reference
front
()
const
{
SPAN_EXPECT
(
!
empty
());
return
*
data
();
}
constexpr
reference
back
()
const
{
SPAN_EXPECT
(
!
empty
());
return
*
(
data
()
+
(
size
()
-
1
));
}
constexpr
pointer
data
()
const
noexcept
{
return
storage_
.
ptr
;
}
// [span.iterators], span iterator support
constexpr
iterator
begin
()
const
noexcept
{
return
data
();
}
constexpr
iterator
end
()
const
noexcept
{
return
data
()
+
size
();
}
constexpr
reverse_iterator
rbegin
()
const
noexcept
{
return
reverse_iterator
(
end
());
}
constexpr
reverse_iterator
rend
()
const
noexcept
{
return
reverse_iterator
(
begin
());
}
private:
storage_type
storage_
{};
};
#ifdef __cpp_deduction_guides
/* Deduction Guides */
template
<
typename
T
,
size_t
N
>
span
(
T
(
&
)[
N
])
->
span
<
T
,
N
>
;
template
<
typename
T
,
size_t
N
>
span
(
std
::
array
<
T
,
N
>&
)
->
span
<
T
,
N
>
;
template
<
typename
T
,
size_t
N
>
span
(
const
std
::
array
<
T
,
N
>&
)
->
span
<
const
T
,
N
>
;
template
<
typename
Container
>
span
(
Container
&
)
->
span
<
typename
std
::
remove_reference
<
// NOLINT
decltype
(
*
detail
::
data
(
std
::
declval
<
Container
&>
()))
>::
type
>
;
template
<
typename
Container
>
span
(
const
Container
&
)
->
span
<
const
typename
Container
::
value_type
>
;
#endif
template
<
typename
ElementType
,
std
::
size_t
Extent
>
span
<
const
detail
::
byte
,
(
Extent
==
dynamic_extent
?
dynamic_extent
:
sizeof
(
ElementType
)
*
Extent
)
>
as_bytes
(
span
<
ElementType
,
Extent
>
s
)
noexcept
{
return
{
reinterpret_cast
<
const
detail
::
byte
*>
(
s
.
data
()),
s
.
size_bytes
()};
}
template
<
typename
ElementType
,
std
::
size_t
Extent
,
typename
std
::
enable_if
<!
std
::
is_const
<
ElementType
>
::
value
,
int
>::
type
=
0
>
span
<
detail
::
byte
,
(
Extent
==
dynamic_extent
?
dynamic_extent
:
sizeof
(
ElementType
)
*
Extent
)
>
as_writable_bytes
(
span
<
ElementType
,
Extent
>
s
)
noexcept
{
return
{
reinterpret_cast
<
detail
::
byte
*>
(
s
.
data
()),
s
.
size_bytes
()};
}
}
// namespace paddle
namespace
std
{
template
<
typename
ElementType
,
std
::
size_t
Extent
>
class
tuple_size
<::
paddle
::
span
<
ElementType
,
Extent
>>
:
public
integral_constant
<
size_t
,
Extent
>
{};
template
<
typename
ElementType
>
class
tuple_size
<
::
paddle
::
span
<
ElementType
,
::
paddle
::
dynamic_extent
>>
;
// not defined
template
<
size_t
I
,
typename
ElementType
,
size_t
Extent
>
class
tuple_element
<
I
,
::
paddle
::
span
<
ElementType
,
Extent
>>
{
public:
static_assert
(
Extent
!=
::
paddle
::
dynamic_extent
&&
I
<
Extent
,
""
);
using
type
=
ElementType
;
};
}
// namespace std
paddle/utils/span_test.cc
0 → 100644
浏览文件 @
3b4bdc08
// Copyright (c) 2023 PaddlePaddle Authors. All Rights Reserved.
// This file copy from https://github.com/tcbrindle/span
// Modified the following points
// 1. remove macros for backward compatibility with pre-C++17 standards
// 2. instantiated namespace name with paddle
/*
This is an implementation of C++20's std::span
http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2019/n4820.pdf
*/
// Copyright Tristan Brindle 2018.
// Distributed under the Boost Software License, Version 1.0.
// (See accompanying file ../../LICENSE_1_0.txt or copy at
// https://www.boost.org/LICENSE_1_0.txt)
#include <cassert>
#include <deque>
#include <initializer_list>
#include <vector>
#include "glog/logging.h"
#include "gtest/gtest.h"
#include "paddle/utils/span.h"
using
paddle
::
span
;
// span();
TEST
(
default_ctor
,
span
)
{
static_assert
(
std
::
is_nothrow_default_constructible
<
span
<
int
>>::
value
,
""
);
static_assert
(
std
::
is_nothrow_default_constructible
<
span
<
int
,
0
>>::
value
,
""
);
static_assert
(
!
std
::
is_default_constructible
<
span
<
int
,
42
>>::
value
,
""
);
// dynamic size
{
constexpr
span
<
int
>
s
{};
static_assert
(
s
.
size
()
==
0
,
""
);
static_assert
(
s
.
data
()
==
nullptr
,
""
);
#ifndef _MSC_VER
static_assert
(
s
.
begin
()
==
s
.
end
(),
""
);
#else
CHECK
(
s
.
begin
()
==
s
.
end
());
#endif
}
// fixed size
{
constexpr
span
<
int
,
0
>
s
{};
static_assert
(
s
.
size
()
==
0
,
""
);
static_assert
(
s
.
data
()
==
nullptr
,
""
);
#ifndef _MSC_VER
static_assert
(
s
.
begin
()
==
s
.
end
(),
""
);
#else
CHECK
(
s
.
begin
()
==
s
.
end
());
#endif
}
}
// span(pointer ptr, size_type count);
TEST
(
pointer_length_ctor
,
span
)
{
static_assert
(
std
::
is_constructible
<
span
<
int
>
,
int
*
,
int
>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
>
,
int
*
,
int
>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
>
,
const
int
*
,
int
>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
int
,
42
>
,
int
*
,
int
>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
,
42
>
,
int
*
,
int
>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
,
42
>
,
const
int
*
,
int
>::
value
,
""
);
// dynamic size
{
int
arr
[]
=
{
1
,
2
,
3
};
span
<
int
>
s
(
arr
,
3
);
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
);
CHECK_EQ
(
s
.
begin
(),
std
::
begin
(
arr
));
CHECK_EQ
(
s
.
end
(),
std
::
end
(
arr
));
}
// fixed size
{
int
arr
[]
=
{
1
,
2
,
3
};
span
<
int
,
3
>
s
(
arr
,
3
);
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
);
CHECK_EQ
(
s
.
begin
(),
std
::
begin
(
arr
));
CHECK_EQ
(
s
.
end
(),
std
::
end
(
arr
));
}
}
// span(pointer ptr, pointer ptr);
TEST
(
pointer_pointer_ctor
,
span
)
{
static_assert
(
std
::
is_constructible
<
span
<
int
>
,
int
*
,
int
*>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
>
,
float
*
,
float
*>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
int
,
42
>
,
int
*
,
int
*>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
42
>
,
float
*
,
float
*>::
value
,
""
);
// dynamic size
{
int
arr
[]
=
{
1
,
2
,
3
};
span
<
int
>
s
{
arr
,
arr
+
3
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
);
CHECK_EQ
(
s
.
begin
(),
std
::
begin
(
arr
));
CHECK_EQ
(
s
.
end
(),
std
::
end
(
arr
));
}
// fixed size
{
int
arr
[]
=
{
1
,
2
,
3
};
span
<
int
,
3
>
s
{
arr
,
arr
+
3
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
);
CHECK_EQ
(
s
.
begin
(),
std
::
begin
(
arr
));
CHECK_EQ
(
s
.
end
(),
std
::
end
(
arr
));
}
}
TEST
(
c_array_ctor
,
span
)
{
using
int_array_t
=
int
[
3
];
using
float_array_t
=
float
[
3
];
static_assert
(
std
::
is_nothrow_constructible
<
span
<
int
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
>
,
float_array_t
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
const
int
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
const
int
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
>
,
float_array_t
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
int
,
3
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
3
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
3
>
,
float_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
const
int
,
3
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
const
int
,
3
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
3
>
,
float_array_t
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
42
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
42
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
42
>
,
float_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
42
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
42
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
42
>
,
float_array_t
&>::
value
,
""
);
// non-const, dynamic size
{
int
arr
[]
=
{
1
,
2
,
3
};
span
<
int
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
);
CHECK_EQ
(
s
.
begin
(),
std
::
begin
(
arr
));
CHECK_EQ
(
s
.
end
(),
std
::
end
(
arr
));
}
// const, dynamic size
{
int
arr
[]
=
{
1
,
2
,
3
};
span
<
int
const
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
);
CHECK_EQ
(
s
.
begin
(),
std
::
begin
(
arr
));
CHECK_EQ
(
s
.
end
(),
std
::
end
(
arr
));
}
// non-const, static size
{
int
arr
[]
=
{
1
,
2
,
3
};
span
<
int
,
3
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
);
CHECK_EQ
(
s
.
begin
(),
std
::
begin
(
arr
));
CHECK_EQ
(
s
.
end
(),
std
::
end
(
arr
));
}
// const, dynamic size
{
int
arr
[]
=
{
1
,
2
,
3
};
span
<
int
const
,
3
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
);
CHECK_EQ
(
s
.
begin
(),
std
::
begin
(
arr
));
CHECK_EQ
(
s
.
end
(),
std
::
end
(
arr
));
}
}
TEST
(
std_array_ctor
,
span
)
{
using
int_array_t
=
std
::
array
<
int
,
3
>
;
using
float_array_t
=
std
::
array
<
float
,
3
>
;
using
zero_array_t
=
std
::
array
<
int
,
0
>
;
static_assert
(
std
::
is_nothrow_constructible
<
span
<
int
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
>
,
float_array_t
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
const
int
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
const
int
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
>
,
float_array_t
const
&>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
int
,
3
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
3
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
3
>
,
float_array_t
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
const
int
,
3
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
span
<
const
int
,
3
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
3
>
,
float_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
42
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
42
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
42
>
,
float_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
42
>
,
int_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
42
>
,
int_array_t
const
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
42
>
,
float_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
int
>
,
zero_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
>
,
const
zero_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
>
,
zero_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
>
,
const
zero_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
int
,
0
>
,
zero_array_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
0
>
,
const
zero_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
,
0
>
,
zero_array_t
&>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
,
0
>
,
const
zero_array_t
&>::
value
,
""
);
// non-const, dynamic size
{
int_array_t
arr
=
{
1
,
2
,
3
};
span
<
int
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
.
data
());
CHECK_EQ
(
s
.
begin
(),
arr
.
data
());
CHECK_EQ
(
s
.
end
(),
arr
.
data
()
+
3
);
}
// const, dynamic size
{
int_array_t
arr
=
{
1
,
2
,
3
};
span
<
int
const
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
.
data
());
CHECK_EQ
(
s
.
begin
(),
arr
.
data
());
CHECK_EQ
(
s
.
end
(),
arr
.
data
()
+
3
);
}
// non-const, static size
{
int_array_t
arr
=
{
1
,
2
,
3
};
span
<
int
,
3
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
.
data
());
CHECK_EQ
(
s
.
begin
(),
arr
.
data
());
CHECK_EQ
(
s
.
end
(),
arr
.
data
()
+
3
);
}
// const, dynamic size
{
int_array_t
arr
=
{
1
,
2
,
3
};
span
<
int
const
,
3
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
.
data
());
CHECK_EQ
(
s
.
begin
(),
arr
.
data
());
CHECK_EQ
(
s
.
end
(),
arr
.
data
()
+
3
);
}
}
TEST
(
ctor_from_containers
,
span
)
{
using
vec_t
=
std
::
vector
<
int
>
;
using
deque_t
=
std
::
deque
<
int
>
;
static_assert
(
std
::
is_constructible
<
span
<
int
>
,
vec_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
>
,
const
vec_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
>
,
const
deque_t
&>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
>
,
vec_t
&>::
value
,
""
);
static_assert
(
std
::
is_constructible
<
span
<
const
int
>
,
const
vec_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
>
,
const
deque_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
3
>
,
vec_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
3
>
,
const
vec_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
int
,
3
>
,
const
deque_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
3
>
,
vec_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
3
>
,
const
vec_t
&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
int
,
3
>
,
const
deque_t
&>::
value
,
""
);
// vector<bool> is not contiguous and cannot be converted to span<bool>
// Regression test for https://github.com/tcbrindle/span/issues/24
static_assert
(
!
std
::
is_constructible
<
span
<
bool
>
,
std
::
vector
<
bool
>&>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
span
<
const
bool
>
,
const
std
::
vector
<
bool
>&>::
value
,
""
);
// non-const, dynamic size
{
vec_t
arr
=
{
1
,
2
,
3
};
span
<
int
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
.
data
());
CHECK_EQ
(
s
.
begin
(),
arr
.
data
());
CHECK_EQ
(
s
.
end
(),
arr
.
data
()
+
3
);
}
// const, dynamic size
{
vec_t
arr
=
{
1
,
2
,
3
};
span
<
int
const
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
.
data
());
CHECK_EQ
(
s
.
begin
(),
arr
.
data
());
CHECK_EQ
(
s
.
end
(),
arr
.
data
()
+
3
);
}
// non-const, static size
{
std
::
array
<
int
,
3
>
arr
=
{
1
,
2
,
3
};
span
<
int
,
3
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
.
data
());
CHECK_EQ
(
s
.
begin
(),
arr
.
data
());
CHECK_EQ
(
s
.
end
(),
arr
.
data
()
+
3
);
}
// const, dynamic size
{
std
::
array
<
int
,
3
>
arr
=
{
1
,
2
,
3
};
span
<
int
const
,
3
>
s
{
arr
};
CHECK_EQ
(
s
.
size
(),
3UL
);
CHECK_EQ
(
s
.
data
(),
arr
.
data
());
CHECK_EQ
(
s
.
begin
(),
arr
.
data
());
CHECK_EQ
(
s
.
end
(),
arr
.
data
()
+
3
);
}
}
TEST
(
ctor_from_spans
,
span
)
{
using
zero_span
=
span
<
int
,
0
>
;
using
zero_const_span
=
span
<
const
int
,
0
>
;
using
big_span
=
span
<
int
,
1000000
>
;
using
big_const_span
=
span
<
const
int
,
1000000
>
;
using
dynamic_span
=
span
<
int
>
;
using
dynamic_const_span
=
span
<
const
int
>
;
static_assert
(
std
::
is_trivially_copyable
<
zero_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_move_constructible
<
zero_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
zero_span
,
zero_const_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
zero_span
,
big_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
zero_span
,
big_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
zero_span
,
dynamic_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
zero_span
,
dynamic_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
zero_const_span
,
zero_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_copyable
<
zero_const_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_move_constructible
<
zero_const_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
zero_const_span
,
big_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
zero_const_span
,
big_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
zero_const_span
,
dynamic_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
zero_const_span
,
dynamic_const_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
big_span
,
zero_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
big_span
,
zero_const_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_copyable
<
big_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_move_constructible
<
big_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
big_span
,
big_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
big_span
,
dynamic_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
big_span
,
dynamic_const_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
big_const_span
,
zero_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
big_const_span
,
zero_const_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_copyable
<
big_const_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_move_constructible
<
big_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
big_const_span
,
big_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
big_const_span
,
dynamic_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
big_const_span
,
dynamic_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
dynamic_span
,
zero_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
dynamic_span
,
zero_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
dynamic_span
,
big_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
dynamic_span
,
big_const_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_copyable
<
dynamic_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_move_constructible
<
dynamic_span
>::
value
,
""
);
static_assert
(
!
std
::
is_constructible
<
dynamic_span
,
dynamic_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
dynamic_const_span
,
zero_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
dynamic_const_span
,
zero_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
dynamic_const_span
,
big_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
dynamic_const_span
,
big_const_span
>::
value
,
""
);
static_assert
(
std
::
is_nothrow_constructible
<
dynamic_const_span
,
dynamic_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_copyable
<
dynamic_const_span
>::
value
,
""
);
static_assert
(
std
::
is_trivially_move_constructible
<
dynamic_const_span
>::
value
,
""
);
constexpr
zero_const_span
s0
{};
constexpr
dynamic_const_span
d
{
s0
};
static_assert
(
d
.
size
()
==
0
,
""
);
static_assert
(
d
.
data
()
==
nullptr
,
""
);
#ifndef _MSC_VER
static_assert
(
d
.
begin
()
==
d
.
end
(),
""
);
#else
CHECK
(
d
.
begin
()
==
d
.
end
());
#endif
}
TEST
(
subview
,
span
)
{
// first<N>
{
int
arr
[]
=
{
1
,
2
,
3
,
4
,
5
};
span
<
int
,
5
>
s
{
arr
};
auto
f
=
s
.
first
<
3
>
();
static_assert
(
std
::
is_same
<
decltype
(
f
),
span
<
int
,
3
>>::
value
,
""
);
CHECK_EQ
(
f
.
size
(),
3UL
);
CHECK_EQ
(
f
.
data
(),
arr
);
CHECK_EQ
(
f
.
begin
(),
arr
);
CHECK_EQ
(
f
.
end
(),
arr
+
3
);
}
// last<N>
{
int
arr
[]
=
{
1
,
2
,
3
,
4
,
5
};
span
<
int
,
5
>
s
{
arr
};
auto
l
=
s
.
last
<
3
>
();
static_assert
(
std
::
is_same
<
decltype
(
l
),
span
<
int
,
3
>>::
value
,
""
);
CHECK_EQ
(
l
.
size
(),
3UL
);
CHECK_EQ
(
l
.
data
(),
arr
+
2
);
CHECK_EQ
(
l
.
begin
(),
arr
+
2
);
CHECK_EQ
(
l
.
end
(),
std
::
end
(
arr
));
}
// subspan<N>
{
int
arr
[]
=
{
1
,
2
,
3
,
4
,
5
};
span
<
int
,
5
>
s
{
arr
};
auto
ss
=
s
.
subspan
<
1
,
2
>
();
static_assert
(
std
::
is_same
<
decltype
(
ss
),
span
<
int
,
2
>>::
value
,
""
);
CHECK_EQ
(
ss
.
size
(),
2UL
);
CHECK_EQ
(
ss
.
data
(),
arr
+
1
);
CHECK_EQ
(
ss
.
begin
(),
arr
+
1
);
CHECK_EQ
(
ss
.
end
(),
arr
+
1
+
2
);
}
// first(n)
{
int
arr
[]
=
{
1
,
2
,
3
,
4
,
5
};
span
<
int
,
5
>
s
{
arr
};
auto
f
=
s
.
first
(
3
);
static_assert
(
std
::
is_same
<
decltype
(
f
),
span
<
int
>>::
value
,
""
);
CHECK_EQ
(
f
.
size
(),
3UL
);
CHECK_EQ
(
f
.
data
(),
arr
);
CHECK_EQ
(
f
.
begin
(),
arr
);
CHECK_EQ
(
f
.
end
(),
arr
+
3
);
}
// last(n)
{
int
arr
[]
=
{
1
,
2
,
3
,
4
,
5
};
span
<
int
,
5
>
s
{
arr
};
auto
l
=
s
.
last
(
3
);
static_assert
(
std
::
is_same
<
decltype
(
l
),
span
<
int
>>::
value
,
""
);
CHECK_EQ
(
l
.
size
(),
3UL
);
CHECK_EQ
(
l
.
data
(),
arr
+
2
);
CHECK_EQ
(
l
.
begin
(),
arr
+
2
);
CHECK_EQ
(
l
.
end
(),
std
::
end
(
arr
));
}
// subspan(n)
{
int
arr
[]
=
{
1
,
2
,
3
,
4
,
5
};
span
<
int
,
5
>
s
{
arr
};
auto
ss
=
s
.
subspan
(
1
,
2
);
static_assert
(
std
::
is_same
<
decltype
(
ss
),
span
<
int
>>::
value
,
""
);
CHECK_EQ
(
ss
.
size
(),
2UL
);
CHECK_EQ
(
ss
.
data
(),
arr
+
1
);
CHECK_EQ
(
ss
.
begin
(),
arr
+
1
);
CHECK_EQ
(
ss
.
end
(),
arr
+
1
+
2
);
}
// TODO(tcbrindle): Test all the dynamic subspan possibilities
}
TEST
(
observers
,
span
)
{
// We already use this everywhere, but whatever
constexpr
span
<
int
,
0
>
empty
{};
static_assert
(
empty
.
size
()
==
0
,
""
);
static_assert
(
empty
.
empty
(),
""
);
constexpr
int
arr
[]
=
{
1
,
2
,
3
};
static_assert
(
span
<
const
int
>
{
arr
}.
size
()
==
3
,
""
);
static_assert
(
!
span
<
const
int
>
{
arr
}.
empty
(),
""
);
}
TEST
(
element_access
,
span
)
{
constexpr
int
arr
[]
=
{
1
,
2
,
3
};
span
<
const
int
>
s
{
arr
};
CHECK_EQ
(
s
[
0
],
arr
[
0
]);
CHECK_EQ
(
s
[
1
],
arr
[
1
]);
CHECK_EQ
(
s
[
2
],
arr
[
2
]);
}
TEST
(
iterator
,
span
)
{
{
std
::
vector
<
int
>
vec
;
span
<
int
>
s
{
vec
};
std
::
sort
(
s
.
begin
(),
s
.
end
());
CHECK
(
std
::
is_sorted
(
vec
.
cbegin
(),
vec
.
cend
()));
}
{
const
std
::
vector
<
int
>
vec
{
1
,
2
,
3
};
span
<
const
int
>
s
{
vec
};
CHECK
(
std
::
equal
(
s
.
rbegin
(),
s
.
rend
(),
vec
.
crbegin
()));
}
}
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录