Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
mariadb
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
mariadb
Commits
d0ede467
Commit
d0ede467
authored
Jan 31, 2006
by
marty@linux.site
Browse files
Options
Browse Files
Download
Plain Diff
Merge mskold@bk-internal.mysql.com:/home/bk/mysql-5.1-new
into linux.site:/home/marty/MySQL/mysql-5.1-new
parents
e9804226
d03efdfd
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
112 additions
and
42 deletions
+112
-42
sql/field.h
sql/field.h
+1
-0
sql/ha_ndbcluster.cc
sql/ha_ndbcluster.cc
+92
-36
sql/ha_ndbcluster.h
sql/ha_ndbcluster.h
+1
-1
sql/sql_table.cc
sql/sql_table.cc
+18
-5
No files found.
sql/field.h
View file @
d0ede467
...
@@ -63,6 +63,7 @@ public:
...
@@ -63,6 +63,7 @@ public:
const
char
**
table_name
,
*
field_name
;
const
char
**
table_name
,
*
field_name
;
LEX_STRING
comment
;
LEX_STRING
comment
;
query_id_t
query_id
;
// For quick test of used fields
query_id_t
query_id
;
// For quick test of used fields
bool
add_index
;
// For check if field will be indexed
/* Field is part of the following keys */
/* Field is part of the following keys */
key_map
key_start
,
part_of_key
,
part_of_sortkey
;
key_map
key_start
,
part_of_key
,
part_of_sortkey
;
/*
/*
...
...
sql/ha_ndbcluster.cc
View file @
d0ede467
...
@@ -1256,10 +1256,9 @@ int ha_ndbcluster::open_indexes(Ndb *ndb, TABLE *tab)
...
@@ -1256,10 +1256,9 @@ int ha_ndbcluster::open_indexes(Ndb *ndb, TABLE *tab)
Renumber indexes in index list by shifting out
Renumber indexes in index list by shifting out
indexes that are to be dropped
indexes that are to be dropped
*/
*/
int
ha_ndbcluster
::
renumber_indexes
(
Ndb
*
ndb
,
TABLE
*
tab
)
void
ha_ndbcluster
::
renumber_indexes
(
Ndb
*
ndb
,
TABLE
*
tab
)
{
{
uint
i
;
uint
i
;
int
error
=
0
;
const
char
*
index_name
;
const
char
*
index_name
;
KEY
*
key_info
=
tab
->
key_info
;
KEY
*
key_info
=
tab
->
key_info
;
const
char
**
key_name
=
tab
->
s
->
keynames
.
type_names
;
const
char
**
key_name
=
tab
->
s
->
keynames
.
type_names
;
...
@@ -1288,7 +1287,7 @@ int ha_ndbcluster::renumber_indexes(Ndb *ndb, TABLE *tab)
...
@@ -1288,7 +1287,7 @@ int ha_ndbcluster::renumber_indexes(Ndb *ndb, TABLE *tab)
}
}
}
}
DBUG_
RETURN
(
error
)
;
DBUG_
VOID_RETURN
;
}
}
/*
/*
...
@@ -4418,7 +4417,7 @@ int ha_ndbcluster::create_handler_files(const char *file)
...
@@ -4418,7 +4417,7 @@ int ha_ndbcluster::create_handler_files(const char *file)
NDBDICT
*
dict
=
ndb
->
getDictionary
();
NDBDICT
*
dict
=
ndb
->
getDictionary
();
if
(
!
(
tab
=
dict
->
getTable
(
m_tabname
)))
if
(
!
(
tab
=
dict
->
getTable
(
m_tabname
)))
DBUG_RETURN
(
0
);
// Must be a create, ignore since frm is saved in create
DBUG_RETURN
(
0
);
// Must be a create, ignore since frm is saved in create
DBUG_ASSERT
(
m_share
->
state
==
NSS_ALTERED
);
name
=
table
->
s
->
normalized_path
.
str
;
name
=
table
->
s
->
normalized_path
.
str
;
DBUG_PRINT
(
"enter"
,
(
"m_tabname: %s, path: %s"
,
m_tabname
,
name
));
DBUG_PRINT
(
"enter"
,
(
"m_tabname: %s, path: %s"
,
m_tabname
,
name
));
if
(
readfrm
(
name
,
&
data
,
&
length
)
||
if
(
readfrm
(
name
,
&
data
,
&
length
)
||
...
@@ -4427,17 +4426,18 @@ int ha_ndbcluster::create_handler_files(const char *file)
...
@@ -4427,17 +4426,18 @@ int ha_ndbcluster::create_handler_files(const char *file)
DBUG_PRINT
(
"info"
,
(
"Missing frm for %s"
,
m_tabname
));
DBUG_PRINT
(
"info"
,
(
"Missing frm for %s"
,
m_tabname
));
my_free
((
char
*
)
data
,
MYF
(
MY_ALLOW_ZERO_PTR
));
my_free
((
char
*
)
data
,
MYF
(
MY_ALLOW_ZERO_PTR
));
my_free
((
char
*
)
pack_data
,
MYF
(
MY_ALLOW_ZERO_PTR
));
my_free
((
char
*
)
pack_data
,
MYF
(
MY_ALLOW_ZERO_PTR
));
DBUG_RETURN
(
1
)
;
error
=
1
;
}
}
if
(
cmp_frm
(
tab
,
pack_data
,
pack_length
))
else
{
{
DBUG_PRINT
(
"info"
,
(
"Table %s has changed, altering frm in ndb"
,
DBUG_PRINT
(
"info"
,
(
"Table %s has changed, altering frm in ndb"
,
m_tabname
));
m_tabname
));
error
=
table_changed
(
pack_data
,
pack_length
);
error
=
table_changed
(
pack_data
,
pack_length
);
m_share
->
state
=
NSS_INITIAL
;
my_free
((
char
*
)
data
,
MYF
(
MY_ALLOW_ZERO_PTR
));
my_free
((
char
*
)
pack_data
,
MYF
(
MY_ALLOW_ZERO_PTR
));
}
}
m
y_free
((
char
*
)
data
,
MYF
(
MY_ALLOW_ZERO_PTR
))
;
m
_share
->
state
=
NSS_INITIAL
;
my_free
((
char
*
)
pack_data
,
MYF
(
MY_ALLOW_ZERO_PTR
));
free_share
(
&
m_share
);
// Decrease ref_count
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
...
@@ -4554,6 +4554,7 @@ int ha_ndbcluster::add_index(TABLE *table_arg,
...
@@ -4554,6 +4554,7 @@ int ha_ndbcluster::add_index(TABLE *table_arg,
int
error
=
0
;
int
error
=
0
;
uint
idx
;
uint
idx
;
DBUG_ASSERT
(
m_share
->
state
==
NSS_INITIAL
);
for
(
idx
=
0
;
idx
<
num_of_keys
;
idx
++
)
for
(
idx
=
0
;
idx
<
num_of_keys
;
idx
++
)
{
{
KEY
*
key
=
key_info
+
idx
;
KEY
*
key
=
key_info
+
idx
;
...
@@ -4569,7 +4570,11 @@ int ha_ndbcluster::add_index(TABLE *table_arg,
...
@@ -4569,7 +4570,11 @@ int ha_ndbcluster::add_index(TABLE *table_arg,
if
((
error
=
create_index
(
key_info
[
idx
].
name
,
key
,
idx_type
,
idx
)))
if
((
error
=
create_index
(
key_info
[
idx
].
name
,
key
,
idx_type
,
idx
)))
break
;
break
;
}
}
m_share
->
state
=
NSS_ALTERED
;
if
(
!
error
)
{
ndbcluster_get_share
(
m_share
);
// Increase ref_count
m_share
->
state
=
NSS_ALTERED
;
}
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
...
@@ -4593,6 +4598,7 @@ int ha_ndbcluster::prepare_drop_index(TABLE *table_arg,
...
@@ -4593,6 +4598,7 @@ int ha_ndbcluster::prepare_drop_index(TABLE *table_arg,
uint
*
key_num
,
uint
num_of_keys
)
uint
*
key_num
,
uint
num_of_keys
)
{
{
DBUG_ENTER
(
"ha_ndbcluster::prepare_drop_index"
);
DBUG_ENTER
(
"ha_ndbcluster::prepare_drop_index"
);
DBUG_ASSERT
(
m_share
->
state
==
NSS_INITIAL
);
// Mark indexes for deletion
// Mark indexes for deletion
uint
idx
;
uint
idx
;
for
(
idx
=
0
;
idx
<
num_of_keys
;
idx
++
)
for
(
idx
=
0
;
idx
<
num_of_keys
;
idx
++
)
...
@@ -4604,8 +4610,10 @@ int ha_ndbcluster::prepare_drop_index(TABLE *table_arg,
...
@@ -4604,8 +4610,10 @@ int ha_ndbcluster::prepare_drop_index(TABLE *table_arg,
THD
*
thd
=
current_thd
;
THD
*
thd
=
current_thd
;
Thd_ndb
*
thd_ndb
=
get_thd_ndb
(
thd
);
Thd_ndb
*
thd_ndb
=
get_thd_ndb
(
thd
);
Ndb
*
ndb
=
thd_ndb
->
ndb
;
Ndb
*
ndb
=
thd_ndb
->
ndb
;
renumber_indexes
(
ndb
,
table_arg
);
ndbcluster_get_share
(
m_share
);
// Increase ref_count
m_share
->
state
=
NSS_ALTERED
;
m_share
->
state
=
NSS_ALTERED
;
DBUG_RETURN
(
renumber_indexes
(
ndb
,
table_arg
)
);
DBUG_RETURN
(
0
);
}
}
/*
/*
...
@@ -4613,13 +4621,19 @@ int ha_ndbcluster::prepare_drop_index(TABLE *table_arg,
...
@@ -4613,13 +4621,19 @@ int ha_ndbcluster::prepare_drop_index(TABLE *table_arg,
*/
*/
int
ha_ndbcluster
::
final_drop_index
(
TABLE
*
table_arg
)
int
ha_ndbcluster
::
final_drop_index
(
TABLE
*
table_arg
)
{
{
int
error
;
DBUG_ENTER
(
"ha_ndbcluster::final_drop_index"
);
DBUG_ENTER
(
"ha_ndbcluster::final_drop_index"
);
DBUG_PRINT
(
"info"
,
(
"ha_ndbcluster::final_drop_index"
));
DBUG_PRINT
(
"info"
,
(
"ha_ndbcluster::final_drop_index"
));
// Really drop indexes
// Really drop indexes
THD
*
thd
=
current_thd
;
THD
*
thd
=
current_thd
;
Thd_ndb
*
thd_ndb
=
get_thd_ndb
(
thd
);
Thd_ndb
*
thd_ndb
=
get_thd_ndb
(
thd
);
Ndb
*
ndb
=
thd_ndb
->
ndb
;
Ndb
*
ndb
=
thd_ndb
->
ndb
;
DBUG_RETURN
(
drop_indexes
(
ndb
,
table_arg
));
if
((
error
=
drop_indexes
(
ndb
,
table_arg
)))
{
m_share
->
state
=
NSS_INITIAL
;
free_share
(
&
m_share
);
// Decrease ref_count
}
DBUG_RETURN
(
error
);
}
}
/*
/*
...
@@ -5237,41 +5251,66 @@ int ndbcluster_discover(THD* thd, const char *db, const char *name,
...
@@ -5237,41 +5251,66 @@ int ndbcluster_discover(THD* thd, const char *db, const char *name,
const
void
*
data
;
const
void
*
data
;
const
NDBTAB
*
tab
;
const
NDBTAB
*
tab
;
Ndb
*
ndb
;
Ndb
*
ndb
;
char
key
[
FN_REFLEN
];
DBUG_ENTER
(
"ndbcluster_discover"
);
DBUG_ENTER
(
"ndbcluster_discover"
);
DBUG_PRINT
(
"enter"
,
(
"db: %s, name: %s"
,
db
,
name
));
DBUG_PRINT
(
"enter"
,
(
"db: %s, name: %s"
,
db
,
name
));
if
(
!
(
ndb
=
check_ndb_in_thd
(
thd
)))
if
(
!
(
ndb
=
check_ndb_in_thd
(
thd
)))
DBUG_RETURN
(
HA_ERR_NO_CONNECTION
);
DBUG_RETURN
(
HA_ERR_NO_CONNECTION
);
ndb
->
setDatabaseName
(
db
);
ndb
->
setDatabaseName
(
db
);
NDBDICT
*
dict
=
ndb
->
getDictionary
();
NDBDICT
*
dict
=
ndb
->
getDictionary
();
dict
->
set_local_table_data_size
(
sizeof
(
Ndb_local_table_statistics
));
dict
->
set_local_table_data_size
(
sizeof
(
Ndb_local_table_statistics
));
dict
->
invalidateTable
(
name
);
dict
->
invalidateTable
(
name
);
if
(
!
(
tab
=
dict
->
getTable
(
name
)))
strxnmov
(
key
,
FN_LEN
-
1
,
mysql_data_home
,
"/"
,
db
,
"/"
,
name
,
NullS
);
{
NDB_SHARE
*
share
=
get_share
(
key
,
0
,
false
);
const
NdbError
err
=
dict
->
getNdbError
();
if
(
share
&&
share
->
state
==
NSS_ALTERED
)
if
(
err
.
code
==
709
||
err
.
code
==
723
)
DBUG_RETURN
(
-
1
);
ERR_RETURN
(
err
);
}
DBUG_PRINT
(
"info"
,
(
"Found table %s"
,
tab
->
getName
()));
len
=
tab
->
getFrmLength
();
if
(
len
==
0
||
tab
->
getFrmData
()
==
NULL
)
{
{
DBUG_PRINT
(
"error"
,
(
"No frm data found."
));
// Frm has been altered on disk, but not yet written to ndb
DBUG_RETURN
(
1
);
if
(
readfrm
(
key
,
&
data
,
&
len
))
{
DBUG_PRINT
(
"error"
,
(
"Could not read frm"
));
if
(
share
)
free_share
(
&
share
);
DBUG_RETURN
(
1
);
}
}
}
else
if
(
unpackfrm
(
&
data
,
&
len
,
tab
->
getFrmData
()))
{
{
DBUG_PRINT
(
"error"
,
(
"Could not unpack table"
));
if
(
!
(
tab
=
dict
->
getTable
(
name
)))
DBUG_RETURN
(
1
);
{
const
NdbError
err
=
dict
->
getNdbError
();
if
(
share
)
free_share
(
&
share
);
if
(
err
.
code
==
709
||
err
.
code
==
723
)
DBUG_RETURN
(
-
1
);
ERR_RETURN
(
err
);
}
DBUG_PRINT
(
"info"
,
(
"Found table %s"
,
tab
->
getName
()));
len
=
tab
->
getFrmLength
();
if
(
len
==
0
||
tab
->
getFrmData
()
==
NULL
)
{
DBUG_PRINT
(
"error"
,
(
"No frm data found."
));
if
(
share
)
free_share
(
&
share
);
DBUG_RETURN
(
1
);
}
if
(
unpackfrm
(
&
data
,
&
len
,
tab
->
getFrmData
()))
{
DBUG_PRINT
(
"error"
,
(
"Could not unpack table"
));
if
(
share
)
free_share
(
&
share
);
DBUG_RETURN
(
1
);
}
}
}
*
frmlen
=
len
;
*
frmlen
=
len
;
*
frmblob
=
data
;
*
frmblob
=
data
;
if
(
share
)
free_share
(
&
share
);
DBUG_RETURN
(
0
);
DBUG_RETURN
(
0
);
}
}
...
@@ -9202,21 +9241,38 @@ uint ha_ndbcluster::set_up_partition_info(partition_info *part_info,
...
@@ -9202,21 +9241,38 @@ uint ha_ndbcluster::set_up_partition_info(partition_info *part_info,
bool
ha_ndbcluster
::
check_if_incompatible_data
(
HA_CREATE_INFO
*
info
,
bool
ha_ndbcluster
::
check_if_incompatible_data
(
HA_CREATE_INFO
*
info
,
uint
table_changes
)
uint
table_changes
)
{
{
return
COMPATIBLE_DATA_NO
;
// Disable fast add/drop index
DBUG_ENTER
(
"ha_ndbcluster::check_if_incompatible_data"
);
uint
i
;
const
NDBTAB
*
tab
=
(
const
NDBTAB
*
)
m_table
;
#ifdef HAVE_NDB_BINLOG
DBUG_PRINT
(
"info"
,
(
"add/drop index not supported with binlog"
));
DBUG_RETURN
(
COMPATIBLE_DATA_NO
);
// Disable fast add/drop index with binlog
#endif
for
(
i
=
0
;
i
<
table
->
s
->
fields
;
i
++
)
{
Field
*
field
=
table
->
field
[
i
];
const
NDBCOL
*
col
=
tab
->
getColumn
(
field
->
field_name
);
if
(
field
->
add_index
&&
col
->
getStorageType
()
==
NdbDictionary
::
Column
::
StorageTypeDisk
)
{
DBUG_PRINT
(
"info"
,
(
"add/drop index not supported for disk stored column"
));
DBUG_RETURN
(
COMPATIBLE_DATA_NO
);
}
}
if
(
table_changes
!=
IS_EQUAL_YES
)
if
(
table_changes
!=
IS_EQUAL_YES
)
return
COMPATIBLE_DATA_NO
;
DBUG_RETURN
(
COMPATIBLE_DATA_NO
)
;
/* Check that auto_increment value was not changed */
/* Check that auto_increment value was not changed */
if
((
info
->
used_fields
&
HA_CREATE_USED_AUTO
)
&&
if
((
info
->
used_fields
&
HA_CREATE_USED_AUTO
)
&&
info
->
auto_increment_value
!=
0
)
info
->
auto_increment_value
!=
0
)
return
COMPATIBLE_DATA_NO
;
DBUG_RETURN
(
COMPATIBLE_DATA_NO
)
;
/* Check that row format didn't change */
/* Check that row format didn't change */
if
((
info
->
used_fields
&
HA_CREATE_USED_AUTO
)
&&
if
((
info
->
used_fields
&
HA_CREATE_USED_AUTO
)
&&
get_row_type
()
!=
info
->
row_type
)
get_row_type
()
!=
info
->
row_type
)
return
COMPATIBLE_DATA_NO
;
DBUG_RETURN
(
COMPATIBLE_DATA_NO
)
;
return
COMPATIBLE_DATA_YES
;
DBUG_RETURN
(
COMPATIBLE_DATA_YES
)
;
}
}
bool
set_up_tablespace
(
st_alter_tablespace
*
info
,
bool
set_up_tablespace
(
st_alter_tablespace
*
info
,
...
...
sql/ha_ndbcluster.h
View file @
d0ede467
...
@@ -685,7 +685,7 @@ private:
...
@@ -685,7 +685,7 @@ private:
void
clear_index
(
int
i
);
void
clear_index
(
int
i
);
void
clear_indexes
();
void
clear_indexes
();
int
open_indexes
(
Ndb
*
ndb
,
TABLE
*
tab
);
int
open_indexes
(
Ndb
*
ndb
,
TABLE
*
tab
);
int
renumber_indexes
(
Ndb
*
ndb
,
TABLE
*
tab
);
void
renumber_indexes
(
Ndb
*
ndb
,
TABLE
*
tab
);
int
drop_indexes
(
Ndb
*
ndb
,
TABLE
*
tab
);
int
drop_indexes
(
Ndb
*
ndb
,
TABLE
*
tab
);
int
add_index_handle
(
THD
*
thd
,
NdbDictionary
::
Dictionary
*
dict
,
int
add_index_handle
(
THD
*
thd
,
NdbDictionary
::
Dictionary
*
dict
,
KEY
*
key_info
,
const
char
*
index_name
,
uint
index_no
);
KEY
*
key_info
,
const
char
*
index_name
,
uint
index_no
);
...
...
sql/sql_table.cc
View file @
d0ede467
...
@@ -3701,12 +3701,10 @@ static uint compare_tables(TABLE *table, List<create_field> *create_list,
...
@@ -3701,12 +3701,10 @@ static uint compare_tables(TABLE *table, List<create_field> *create_list,
/* Evaluate changes bitmap and send to check_if_incompatible_data() */
/* Evaluate changes bitmap and send to check_if_incompatible_data() */
if
(
!
(
tmp
=
field
->
is_equal
(
new_field
)))
if
(
!
(
tmp
=
field
->
is_equal
(
new_field
)))
DBUG_RETURN
(
ALTER_TABLE_DATA_CHANGED
);
DBUG_RETURN
(
ALTER_TABLE_DATA_CHANGED
);
// Clear indexed marker
field
->
add_index
=
0
;
changes
|=
tmp
;
changes
|=
tmp
;
}
}
/* Check if changes are compatible with current handler without a copy */
if
(
table
->
file
->
check_if_incompatible_data
(
create_info
,
changes
))
DBUG_RETURN
(
ALTER_TABLE_DATA_CHANGED
);
/*
/*
Go through keys and check if the original ones are compatible
Go through keys and check if the original ones are compatible
...
@@ -3774,6 +3772,11 @@ static uint compare_tables(TABLE *table, List<create_field> *create_list,
...
@@ -3774,6 +3772,11 @@ static uint compare_tables(TABLE *table, List<create_field> *create_list,
/* Key modified. Add the offset of the key to both buffers. */
/* Key modified. Add the offset of the key to both buffers. */
index_drop_buffer
[(
*
index_drop_count
)
++
]
=
table_key
-
table
->
key_info
;
index_drop_buffer
[(
*
index_drop_count
)
++
]
=
table_key
-
table
->
key_info
;
index_add_buffer
[(
*
index_add_count
)
++
]
=
new_key
-
key_info_buffer
;
index_add_buffer
[(
*
index_add_count
)
++
]
=
new_key
-
key_info_buffer
;
field
=
table
->
field
[
new_key
->
key_part
->
fieldnr
];
// Add field to the key
new_key
->
key_part
->
field
=
table
->
field
[
new_key
->
key_part
->
fieldnr
];
// Mark field to be part of new key
field
->
add_index
=
1
;
DBUG_PRINT
(
"info"
,
(
"index changed: '%s'"
,
table_key
->
name
));
DBUG_PRINT
(
"info"
,
(
"index changed: '%s'"
,
table_key
->
name
));
}
}
/*end of for (; table_key < table_key_end;) */
/*end of for (; table_key < table_key_end;) */
...
@@ -3793,9 +3796,19 @@ static uint compare_tables(TABLE *table, List<create_field> *create_list,
...
@@ -3793,9 +3796,19 @@ static uint compare_tables(TABLE *table, List<create_field> *create_list,
{
{
/* Key not found. Add the offset of the key to the add buffer. */
/* Key not found. Add the offset of the key to the add buffer. */
index_add_buffer
[(
*
index_add_count
)
++
]
=
new_key
-
key_info_buffer
;
index_add_buffer
[(
*
index_add_count
)
++
]
=
new_key
-
key_info_buffer
;
DBUG_PRINT
(
"info"
,
(
"index added: '%s'"
,
new_key
->
name
));
field
=
table
->
field
[
new_key
->
key_part
->
fieldnr
];
// Add field to the key
new_key
->
key_part
->
field
=
table
->
field
[
new_key
->
key_part
->
fieldnr
];
// Mark field to be part of new key
field
->
add_index
=
1
;
DBUG_PRINT
(
"info"
,
(
"index added: '%s'"
,
new_key
->
name
));
}
}
}
}
/* Check if changes are compatible with current handler without a copy */
if
(
table
->
file
->
check_if_incompatible_data
(
create_info
,
changes
))
DBUG_RETURN
(
ALTER_TABLE_DATA_CHANGED
);
if
(
*
index_drop_count
||
*
index_add_count
)
if
(
*
index_drop_count
||
*
index_add_count
)
DBUG_RETURN
(
ALTER_TABLE_INDEX_CHANGED
);
DBUG_RETURN
(
ALTER_TABLE_INDEX_CHANGED
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment