Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
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
nexedi
linux
Commits
22bf2be7
Commit
22bf2be7
authored
Sep 30, 2002
by
Jon Grimm
Browse files
Options
Browse Files
Download
Plain Diff
Merge touki.austin.ibm.com:/home/jgrimm/bk/linux-2.5
into touki.austin.ibm.com:/home/jgrimm/bk/lksctp-2.5.work
parents
7570df54
27cdae33
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
92 additions
and
77 deletions
+92
-77
include/net/sctp/sctp.h
include/net/sctp/sctp.h
+8
-0
net/sctp/protocol.c
net/sctp/protocol.c
+2
-4
net/sctp/socket.c
net/sctp/socket.c
+82
-73
No files found.
include/net/sctp/sctp.h
View file @
22bf2be7
...
@@ -103,6 +103,14 @@
...
@@ -103,6 +103,14 @@
#define SCTP_PROTOSW_FLAG INET_PROTOSW_PERMANENT
#define SCTP_PROTOSW_FLAG INET_PROTOSW_PERMANENT
#endif
#endif
/* Certain internal static functions need to be exported when
* compiled into the test frame.
*/
#ifndef SCTP_STATIC
#define SCTP_STATIC static
#endif
/*
/*
* Function declarations.
* Function declarations.
*/
*/
...
...
net/sctp/protocol.c
View file @
22bf2be7
...
@@ -119,8 +119,7 @@ static inline void sctp_v4_get_local_addr_list(sctp_protocol_t *proto,
...
@@ -119,8 +119,7 @@ static inline void sctp_v4_get_local_addr_list(sctp_protocol_t *proto,
for
(
ifa
=
in_dev
->
ifa_list
;
ifa
;
ifa
=
ifa
->
ifa_next
)
{
for
(
ifa
=
in_dev
->
ifa_list
;
ifa
;
ifa
=
ifa
->
ifa_next
)
{
/* Add the address to the local list. */
/* Add the address to the local list. */
/* XXX BUG: sleeping allocation with lock held -DaveM */
addr
=
t_new
(
struct
sockaddr_storage_list
,
GFP_ATOMIC
);
addr
=
t_new
(
struct
sockaddr_storage_list
,
GFP_KERNEL
);
if
(
addr
)
{
if
(
addr
)
{
INIT_LIST_HEAD
(
&
addr
->
list
);
INIT_LIST_HEAD
(
&
addr
->
list
);
addr
->
a
.
v4
.
sin_family
=
AF_INET
;
addr
->
a
.
v4
.
sin_family
=
AF_INET
;
...
@@ -157,8 +156,7 @@ static inline void sctp_v6_get_local_addr_list(sctp_protocol_t *proto,
...
@@ -157,8 +156,7 @@ static inline void sctp_v6_get_local_addr_list(sctp_protocol_t *proto,
read_lock_bh
(
&
in6_dev
->
lock
);
read_lock_bh
(
&
in6_dev
->
lock
);
for
(
ifp
=
in6_dev
->
addr_list
;
ifp
;
ifp
=
ifp
->
if_next
)
{
for
(
ifp
=
in6_dev
->
addr_list
;
ifp
;
ifp
=
ifp
->
if_next
)
{
/* Add the address to the local list. */
/* Add the address to the local list. */
/* XXX BUG: sleeping allocation with lock held -DaveM */
addr
=
t_new
(
struct
sockaddr_storage_list
,
GFP_ATOMIC
);
addr
=
t_new
(
struct
sockaddr_storage_list
,
GFP_KERNEL
);
if
(
addr
)
{
if
(
addr
)
{
addr
->
a
.
v6
.
sin6_family
=
AF_INET6
;
addr
->
a
.
v6
.
sin6_family
=
AF_INET6
;
addr
->
a
.
v6
.
sin6_port
=
0
;
addr
->
a
.
v6
.
sin6_port
=
0
;
...
...
net/sctp/socket.c
View file @
22bf2be7
...
@@ -73,8 +73,12 @@
...
@@ -73,8 +73,12 @@
#include <net/sock.h>
#include <net/sock.h>
#include <net/sctp/sctp.h>
#include <net/sctp/sctp.h>
/* WARNING: Please do not remove the SCTP_STATIC attribute to
* any of the functions below as they are used to export functions
* used by a project regression testsuite.
*/
/* Forward declarations for internal helper functions. */
/* Forward declarations for internal helper functions. */
static
void
__sctp_write_space
(
sctp_association_t
*
asoc
);
static
int
sctp_writeable
(
struct
sock
*
sk
);
static
int
sctp_writeable
(
struct
sock
*
sk
);
static
inline
int
sctp_wspace
(
sctp_association_t
*
asoc
);
static
inline
int
sctp_wspace
(
sctp_association_t
*
asoc
);
static
inline
void
sctp_set_owner_w
(
sctp_chunk_t
*
chunk
);
static
inline
void
sctp_set_owner_w
(
sctp_chunk_t
*
chunk
);
...
@@ -92,8 +96,7 @@ static int sctp_bindx_add(struct sock *, struct sockaddr_storage *, int);
...
@@ -92,8 +96,7 @@ static int sctp_bindx_add(struct sock *, struct sockaddr_storage *, int);
static
int
sctp_bindx_rem
(
struct
sock
*
,
struct
sockaddr_storage
*
,
int
);
static
int
sctp_bindx_rem
(
struct
sock
*
,
struct
sockaddr_storage
*
,
int
);
static
int
sctp_do_bind
(
struct
sock
*
,
sockaddr_storage_t
*
,
int
);
static
int
sctp_do_bind
(
struct
sock
*
,
sockaddr_storage_t
*
,
int
);
static
int
sctp_autobind
(
struct
sock
*
sk
);
static
int
sctp_autobind
(
struct
sock
*
sk
);
static
sctp_bind_bucket_t
*
sctp_bucket_create
(
sctp_bind_hashbucket_t
*
head
,
unsigned
short
snum
);
/* API 3.1.2 bind() - UDP Style Syntax
/* API 3.1.2 bind() - UDP Style Syntax
* The syntax of bind() is,
* The syntax of bind() is,
...
@@ -132,7 +135,8 @@ int sctp_bind(struct sock *sk, struct sockaddr *uaddr, int addr_len)
...
@@ -132,7 +135,8 @@ int sctp_bind(struct sock *sk, struct sockaddr *uaddr, int addr_len)
static
long
sctp_get_port_local
(
struct
sock
*
,
unsigned
short
);
static
long
sctp_get_port_local
(
struct
sock
*
,
unsigned
short
);
/* Bind a local address either to an endpoint or to an association. */
/* Bind a local address either to an endpoint or to an association. */
static
int
sctp_do_bind
(
struct
sock
*
sk
,
sockaddr_storage_t
*
newaddr
,
int
addr_len
)
SCTP_STATIC
int
sctp_do_bind
(
struct
sock
*
sk
,
sockaddr_storage_t
*
newaddr
,
int
addr_len
)
{
{
sctp_opt_t
*
sp
=
sctp_sk
(
sk
);
sctp_opt_t
*
sp
=
sctp_sk
(
sk
);
sctp_endpoint_t
*
ep
=
sp
->
ep
;
sctp_endpoint_t
*
ep
=
sp
->
ep
;
...
@@ -603,7 +607,7 @@ int sctp_bindx_rem(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt)
...
@@ -603,7 +607,7 @@ int sctp_bindx_rem(struct sock *sk, struct sockaddr_storage *addrs, int addrcnt)
*
*
* Returns 0 if ok, <0 errno code on error.
* Returns 0 if ok, <0 errno code on error.
*/
*/
static
int
sctp_setsockopt_bindx
(
struct
sock
*
sk
,
SCTP_STATIC
int
sctp_setsockopt_bindx
(
struct
sock
*
sk
,
struct
sockaddr_storage
*
addrs
,
struct
sockaddr_storage
*
addrs
,
int
addrssize
,
int
op
)
int
addrssize
,
int
op
)
{
{
...
@@ -659,7 +663,7 @@ static int sctp_setsockopt_bindx(struct sock* sk,
...
@@ -659,7 +663,7 @@ static int sctp_setsockopt_bindx(struct sock* sk,
* If sd in the close() call is a branched-off socket representing only
* If sd in the close() call is a branched-off socket representing only
* one association, the shutdown is performed on that association only.
* one association, the shutdown is performed on that association only.
*/
*/
static
void
sctp_close
(
struct
sock
*
sk
,
long
timeout
)
SCTP_STATIC
void
sctp_close
(
struct
sock
*
sk
,
long
timeout
)
{
{
sctp_endpoint_t
*
ep
;
sctp_endpoint_t
*
ep
;
sctp_association_t
*
asoc
;
sctp_association_t
*
asoc
;
...
@@ -727,9 +731,9 @@ static void sctp_close(struct sock *sk, long timeout)
...
@@ -727,9 +731,9 @@ static void sctp_close(struct sock *sk, long timeout)
/* BUG: We do not implement timeouts. */
/* BUG: We do not implement timeouts. */
/* BUG: We do not implement the equivalent of wait_for_tcp_memory(). */
/* BUG: We do not implement the equivalent of wait_for_tcp_memory(). */
static
int
sctp_msghdr_parse
(
const
struct
msghdr
*
,
sctp_cmsgs_t
*
);
SCTP_STATIC
int
sctp_msghdr_parse
(
const
struct
msghdr
*
,
sctp_cmsgs_t
*
);
static
int
sctp_sendmsg
(
struct
sock
*
sk
,
struct
msghdr
*
msg
,
int
size
)
SCTP_STATIC
int
sctp_sendmsg
(
struct
sock
*
sk
,
struct
msghdr
*
msg
,
int
size
)
{
{
sctp_opt_t
*
sp
;
sctp_opt_t
*
sp
;
sctp_endpoint_t
*
ep
;
sctp_endpoint_t
*
ep
;
...
@@ -1094,7 +1098,7 @@ static int sctp_sendmsg(struct sock *sk, struct msghdr *msg, int size)
...
@@ -1094,7 +1098,7 @@ static int sctp_sendmsg(struct sock *sk, struct msghdr *msg, int size)
*/
*/
static
struct
sk_buff
*
sctp_skb_recv_datagram
(
struct
sock
*
,
int
,
int
,
int
*
);
static
struct
sk_buff
*
sctp_skb_recv_datagram
(
struct
sock
*
,
int
,
int
,
int
*
);
static
int
sctp_recvmsg
(
struct
sock
*
sk
,
struct
msghdr
*
msg
,
int
len
,
SCTP_STATIC
int
sctp_recvmsg
(
struct
sock
*
sk
,
struct
msghdr
*
msg
,
int
len
,
int
noblock
,
int
flags
,
int
*
addr_len
)
int
noblock
,
int
flags
,
int
*
addr_len
)
{
{
sctp_ulpevent_t
*
event
=
NULL
;
sctp_ulpevent_t
*
event
=
NULL
;
...
@@ -1229,7 +1233,7 @@ static inline int sctp_setsockopt_autoclose(struct sock *sk, char *optval,
...
@@ -1229,7 +1233,7 @@ static inline int sctp_setsockopt_autoclose(struct sock *sk, char *optval,
* optval - the buffer to store the value of the option.
* optval - the buffer to store the value of the option.
* optlen - the size of the buffer.
* optlen - the size of the buffer.
*/
*/
static
int
sctp_setsockopt
(
struct
sock
*
sk
,
int
level
,
int
optname
,
SCTP_STATIC
int
sctp_setsockopt
(
struct
sock
*
sk
,
int
level
,
int
optname
,
char
*
optval
,
int
optlen
)
char
*
optval
,
int
optlen
)
{
{
int
retval
=
0
;
int
retval
=
0
;
...
@@ -1313,19 +1317,20 @@ static int sctp_setsockopt(struct sock *sk, int level, int optname,
...
@@ -1313,19 +1317,20 @@ static int sctp_setsockopt(struct sock *sk, int level, int optname,
}
}
/* FIXME: Write comments. */
/* FIXME: Write comments. */
static
int
sctp_connect
(
struct
sock
*
sk
,
struct
sockaddr
*
uaddr
,
int
addr_len
)
SCTP_STATIC
int
sctp_connect
(
struct
sock
*
sk
,
struct
sockaddr
*
uaddr
,
int
addr_len
)
{
{
return
-
EOPNOTSUPP
;
/* STUB */
return
-
EOPNOTSUPP
;
/* STUB */
}
}
/* FIXME: Write comments. */
/* FIXME: Write comments. */
static
int
sctp_disconnect
(
struct
sock
*
sk
,
int
flags
)
SCTP_STATIC
int
sctp_disconnect
(
struct
sock
*
sk
,
int
flags
)
{
{
return
-
EOPNOTSUPP
;
/* STUB */
return
-
EOPNOTSUPP
;
/* STUB */
}
}
/* FIXME: Write comments. */
/* FIXME: Write comments. */
static
struct
sock
*
sctp_accept
(
struct
sock
*
sk
,
int
flags
,
int
*
err
)
SCTP_STATIC
struct
sock
*
sctp_accept
(
struct
sock
*
sk
,
int
flags
,
int
*
err
)
{
{
int
error
=
-
EOPNOTSUPP
;
int
error
=
-
EOPNOTSUPP
;
...
@@ -1334,7 +1339,7 @@ static struct sock *sctp_accept(struct sock *sk, int flags, int *err)
...
@@ -1334,7 +1339,7 @@ static struct sock *sctp_accept(struct sock *sk, int flags, int *err)
}
}
/* FIXME: Write Comments. */
/* FIXME: Write Comments. */
static
int
sctp_ioctl
(
struct
sock
*
sk
,
int
cmd
,
unsigned
long
arg
)
SCTP_STATIC
int
sctp_ioctl
(
struct
sock
*
sk
,
int
cmd
,
unsigned
long
arg
)
{
{
return
-
EOPNOTSUPP
;
/* STUB */
return
-
EOPNOTSUPP
;
/* STUB */
}
}
...
@@ -1343,7 +1348,7 @@ static int sctp_ioctl(struct sock *sk, int cmd, unsigned long arg)
...
@@ -1343,7 +1348,7 @@ static int sctp_ioctl(struct sock *sk, int cmd, unsigned long arg)
* initialized the SCTP-specific portion of the sock.
* initialized the SCTP-specific portion of the sock.
* The sock structure should already be zero-filled memory.
* The sock structure should already be zero-filled memory.
*/
*/
static
int
sctp_init_sock
(
struct
sock
*
sk
)
SCTP_STATIC
int
sctp_init_sock
(
struct
sock
*
sk
)
{
{
sctp_endpoint_t
*
ep
;
sctp_endpoint_t
*
ep
;
sctp_protocol_t
*
proto
;
sctp_protocol_t
*
proto
;
...
@@ -1428,7 +1433,7 @@ static int sctp_init_sock(struct sock *sk)
...
@@ -1428,7 +1433,7 @@ static int sctp_init_sock(struct sock *sk)
}
}
/* Cleanup any SCTP per socket resources. */
/* Cleanup any SCTP per socket resources. */
static
int
sctp_destroy_sock
(
struct
sock
*
sk
)
SCTP_STATIC
int
sctp_destroy_sock
(
struct
sock
*
sk
)
{
{
sctp_endpoint_t
*
ep
;
sctp_endpoint_t
*
ep
;
...
@@ -1442,7 +1447,7 @@ static int sctp_destroy_sock(struct sock *sk)
...
@@ -1442,7 +1447,7 @@ static int sctp_destroy_sock(struct sock *sk)
}
}
/* FIXME: Comments needed. */
/* FIXME: Comments needed. */
static
void
sctp_shutdown
(
struct
sock
*
sk
,
int
how
)
SCTP_STATIC
void
sctp_shutdown
(
struct
sock
*
sk
,
int
how
)
{
{
/* UDP-style sockets do not support shutdown. */
/* UDP-style sockets do not support shutdown. */
/* STUB */
/* STUB */
...
@@ -1563,7 +1568,7 @@ static inline int sctp_getsockopt_autoclose(struct sock *sk, int len, char *optv
...
@@ -1563,7 +1568,7 @@ static inline int sctp_getsockopt_autoclose(struct sock *sk, int len, char *optv
}
}
/* Helper routine to branch off an association to a new socket. */
/* Helper routine to branch off an association to a new socket. */
static
int
sctp_do_peeloff
(
sctp_association_t
*
assoc
,
struct
socket
**
newsock
)
SCTP_STATIC
int
sctp_do_peeloff
(
sctp_association_t
*
assoc
,
struct
socket
**
newsock
)
{
{
struct
sock
*
oldsk
=
assoc
->
base
.
sk
;
struct
sock
*
oldsk
=
assoc
->
base
.
sk
;
struct
sock
*
newsk
;
struct
sock
*
newsk
;
...
@@ -1652,7 +1657,7 @@ static inline int sctp_getsockopt_peeloff(struct sock *sk, int len, char *optval
...
@@ -1652,7 +1657,7 @@ static inline int sctp_getsockopt_peeloff(struct sock *sk, int len, char *optval
return
0
;
return
0
;
}
}
static
int
sctp_getsockopt
(
struct
sock
*
sk
,
int
level
,
int
optname
,
SCTP_STATIC
int
sctp_getsockopt
(
struct
sock
*
sk
,
int
level
,
int
optname
,
char
*
optval
,
int
*
optlen
)
char
*
optval
,
int
*
optlen
)
{
{
int
retval
=
0
;
int
retval
=
0
;
...
@@ -1734,6 +1739,8 @@ static void sctp_unhash(struct sock *sk)
...
@@ -1734,6 +1739,8 @@ static void sctp_unhash(struct sock *sk)
* link to the socket (struct sock) that uses it, the port number and
* link to the socket (struct sock) that uses it, the port number and
* a fastreuse flag (FIXME: NPI ipg).
* a fastreuse flag (FIXME: NPI ipg).
*/
*/
static
sctp_bind_bucket_t
*
sctp_bucket_create
(
sctp_bind_hashbucket_t
*
head
,
unsigned
short
snum
);
static
long
sctp_get_port_local
(
struct
sock
*
sk
,
unsigned
short
snum
)
static
long
sctp_get_port_local
(
struct
sock
*
sk
,
unsigned
short
snum
)
{
{
sctp_bind_hashbucket_t
*
head
;
/* hash list */
sctp_bind_hashbucket_t
*
head
;
/* hash list */
...
@@ -1927,7 +1934,7 @@ static int sctp_get_port(struct sock *sk, unsigned short snum)
...
@@ -1927,7 +1934,7 @@ static int sctp_get_port(struct sock *sk, unsigned short snum)
* An application uses listen() to mark a socket as being able to
* An application uses listen() to mark a socket as being able to
* accept new associations.
* accept new associations.
*/
*/
static
int
sctp_seqpacket_listen
(
struct
sock
*
sk
,
int
backlog
)
SCTP_STATIC
int
sctp_seqpacket_listen
(
struct
sock
*
sk
,
int
backlog
)
{
{
sctp_opt_t
*
sp
=
sctp_sk
(
sk
);
sctp_opt_t
*
sp
=
sctp_sk
(
sk
);
sctp_endpoint_t
*
ep
=
sp
->
ep
;
sctp_endpoint_t
*
ep
=
sp
->
ep
;
...
@@ -2184,7 +2191,8 @@ static int sctp_autobind(struct sock *sk)
...
@@ -2184,7 +2191,8 @@ static int sctp_autobind(struct sock *sk)
* msg_control
* msg_control
* points here
* points here
*/
*/
static
int
sctp_msghdr_parse
(
const
struct
msghdr
*
msg
,
sctp_cmsgs_t
*
cmsgs
)
SCTP_STATIC
int
sctp_msghdr_parse
(
const
struct
msghdr
*
msg
,
sctp_cmsgs_t
*
cmsgs
)
{
{
struct
cmsghdr
*
cmsg
;
struct
cmsghdr
*
cmsg
;
...
@@ -2500,6 +2508,31 @@ static inline void sctp_set_owner_w(sctp_chunk_t *chunk)
...
@@ -2500,6 +2508,31 @@ static inline void sctp_set_owner_w(sctp_chunk_t *chunk)
sk
->
wmem_queued
+=
SCTP_DATA_SNDSIZE
(
chunk
);
sk
->
wmem_queued
+=
SCTP_DATA_SNDSIZE
(
chunk
);
}
}
/* If sndbuf has changed, wake up per association sndbuf waiters. */
static
void
__sctp_write_space
(
sctp_association_t
*
asoc
)
{
struct
sock
*
sk
=
asoc
->
base
.
sk
;
struct
socket
*
sock
=
sk
->
socket
;
if
((
sctp_wspace
(
asoc
)
>
0
)
&&
sock
)
{
if
(
waitqueue_active
(
&
asoc
->
wait
))
wake_up_interruptible
(
&
asoc
->
wait
);
if
(
sctp_writeable
(
sk
))
{
if
(
sk
->
sleep
&&
waitqueue_active
(
sk
->
sleep
))
wake_up_interruptible
(
sk
->
sleep
);
/* Note that we try to include the Async I/O support
* here by modeling from the current TCP/UDP code.
* We have not tested with it yet.
*/
if
(
sock
->
fasync_list
&&
!
(
sk
->
shutdown
&
SEND_SHUTDOWN
))
sock_wake_async
(
sock
,
2
,
POLL_OUT
);
}
}
}
/* Do accounting for the sndbuf space.
/* Do accounting for the sndbuf space.
* Decrement the used sndbuf space of the corresponding association by the
* Decrement the used sndbuf space of the corresponding association by the
* data size which was just transmitted(freed).
* data size which was just transmitted(freed).
...
@@ -2522,7 +2555,8 @@ static void sctp_wfree(struct sk_buff *skb)
...
@@ -2522,7 +2555,8 @@ static void sctp_wfree(struct sk_buff *skb)
}
}
/* Helper function to wait for space in the sndbuf. */
/* Helper function to wait for space in the sndbuf. */
static
int
sctp_wait_for_sndbuf
(
sctp_association_t
*
asoc
,
long
*
timeo_p
,
int
msg_len
)
static
int
sctp_wait_for_sndbuf
(
sctp_association_t
*
asoc
,
long
*
timeo_p
,
int
msg_len
)
{
{
struct
sock
*
sk
=
asoc
->
base
.
sk
;
struct
sock
*
sk
=
asoc
->
base
.
sk
;
int
err
=
0
;
int
err
=
0
;
...
@@ -2581,31 +2615,6 @@ static int sctp_wait_for_sndbuf(sctp_association_t *asoc, long *timeo_p, int msg
...
@@ -2581,31 +2615,6 @@ static int sctp_wait_for_sndbuf(sctp_association_t *asoc, long *timeo_p, int msg
goto
out
;
goto
out
;
}
}
/* If sndbuf has changed, wake up per association sndbuf waiters. */
static
void
__sctp_write_space
(
sctp_association_t
*
asoc
)
{
struct
sock
*
sk
=
asoc
->
base
.
sk
;
struct
socket
*
sock
=
sk
->
socket
;
if
((
sctp_wspace
(
asoc
)
>
0
)
&&
sock
)
{
if
(
waitqueue_active
(
&
asoc
->
wait
))
wake_up_interruptible
(
&
asoc
->
wait
);
if
(
sctp_writeable
(
sk
))
{
if
(
sk
->
sleep
&&
waitqueue_active
(
sk
->
sleep
))
wake_up_interruptible
(
sk
->
sleep
);
/* Note that we try to include the Async I/O support
* here by modeling from the current TCP/UDP code.
* We have not tested with it yet.
*/
if
(
sock
->
fasync_list
&&
!
(
sk
->
shutdown
&
SEND_SHUTDOWN
))
sock_wake_async
(
sock
,
2
,
POLL_OUT
);
}
}
}
/* If socket sndbuf has changed, wake up all per association waiters. */
/* If socket sndbuf has changed, wake up all per association waiters. */
void
sctp_write_space
(
struct
sock
*
sk
)
void
sctp_write_space
(
struct
sock
*
sk
)
{
{
...
...
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