Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-cw
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
wine
wine-cw
Commits
10e6f57f
Commit
10e6f57f
authored
Jun 03, 2004
by
Alexandre Julliard
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Set the WINEPRELOADRESERVE variable when starting a new process.
parent
dc61a30b
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
50 additions
and
35 deletions
+50
-35
module.c
dlls/kernel/module.c
+21
-16
process.c
dlls/kernel/process.c
+28
-18
module.h
include/module.h
+1
-1
No files found.
dlls/kernel/module.c
View file @
10e6f57f
...
@@ -130,7 +130,7 @@ good:
...
@@ -130,7 +130,7 @@ good:
/***********************************************************************
/***********************************************************************
* MODULE_GetBinaryType
* MODULE_GetBinaryType
*/
*/
enum
binary_type
MODULE_GetBinaryType
(
HANDLE
hfile
)
enum
binary_type
MODULE_GetBinaryType
(
HANDLE
hfile
,
void
**
res_start
,
void
**
res_end
)
{
{
union
union
{
{
...
@@ -150,7 +150,6 @@ enum binary_type MODULE_GetBinaryType( HANDLE hfile )
...
@@ -150,7 +150,6 @@ enum binary_type MODULE_GetBinaryType( HANDLE hfile )
IMAGE_DOS_HEADER
mz
;
IMAGE_DOS_HEADER
mz
;
}
header
;
}
header
;
char
magic
[
4
];
DWORD
len
;
DWORD
len
;
/* Seek to the start of the file and read the header information. */
/* Seek to the start of the file and read the header information. */
...
@@ -184,6 +183,12 @@ enum binary_type MODULE_GetBinaryType( HANDLE hfile )
...
@@ -184,6 +183,12 @@ enum binary_type MODULE_GetBinaryType( HANDLE hfile )
if
(
header
.
mz
.
e_magic
==
IMAGE_DOS_SIGNATURE
)
if
(
header
.
mz
.
e_magic
==
IMAGE_DOS_SIGNATURE
)
{
{
union
{
IMAGE_OS2_HEADER
os2
;
IMAGE_NT_HEADERS
nt
;
}
ext_header
;
/* We do have a DOS image so we will now try to seek into
/* We do have a DOS image so we will now try to seek into
* the file by the amount indicated by the field
* the file by the amount indicated by the field
* "Offset to extended header" and read in the
* "Offset to extended header" and read in the
...
@@ -193,41 +198,41 @@ enum binary_type MODULE_GetBinaryType( HANDLE hfile )
...
@@ -193,41 +198,41 @@ enum binary_type MODULE_GetBinaryType( HANDLE hfile )
*/
*/
if
(
SetFilePointer
(
hfile
,
header
.
mz
.
e_lfanew
,
NULL
,
SEEK_SET
)
==
-
1
)
if
(
SetFilePointer
(
hfile
,
header
.
mz
.
e_lfanew
,
NULL
,
SEEK_SET
)
==
-
1
)
return
BINARY_DOS
;
return
BINARY_DOS
;
if
(
!
ReadFile
(
hfile
,
magic
,
sizeof
(
magic
),
&
len
,
NULL
)
||
len
!=
sizeof
(
magic
)
)
if
(
!
ReadFile
(
hfile
,
&
ext_header
,
sizeof
(
ext_header
),
&
len
,
NULL
)
||
len
<
4
)
return
BINARY_DOS
;
return
BINARY_DOS
;
/* Reading the magic field succeeded so
/* Reading the magic field succeeded so
* we will try to determine what type it is.
* we will try to determine what type it is.
*/
*/
if
(
!
memcmp
(
magic
,
"PE
\0\0
"
,
4
))
if
(
!
memcmp
(
&
ext_header
.
nt
.
Signature
,
"PE
\0\0
"
,
4
))
{
{
IMAGE_FILE_HEADER
FileHeader
;
if
(
len
>=
sizeof
(
ext_header
.
nt
.
FileHeader
))
if
(
ReadFile
(
hfile
,
&
FileHeader
,
sizeof
(
FileHeader
),
&
len
,
NULL
)
&&
len
==
sizeof
(
FileHeader
))
{
{
if
(
FileHeader
.
Characteristics
&
IMAGE_FILE_DLL
)
return
BINARY_PE_DLL
;
if
(
len
<
sizeof
(
ext_header
.
nt
))
/* clear remaining part of header if missing */
memset
(
(
char
*
)
&
ext_header
.
nt
+
len
,
0
,
sizeof
(
ext_header
.
nt
)
-
len
);
if
(
res_start
)
*
res_start
=
(
void
*
)
ext_header
.
nt
.
OptionalHeader
.
ImageBase
;
if
(
res_end
)
*
res_end
=
(
void
*
)(
ext_header
.
nt
.
OptionalHeader
.
ImageBase
+
ext_header
.
nt
.
OptionalHeader
.
SizeOfImage
);
if
(
ext_header
.
nt
.
FileHeader
.
Characteristics
&
IMAGE_FILE_DLL
)
return
BINARY_PE_DLL
;
return
BINARY_PE_EXE
;
return
BINARY_PE_EXE
;
}
}
return
BINARY_DOS
;
return
BINARY_DOS
;
}
}
if
(
!
memcmp
(
magic
,
"NE"
,
2
))
if
(
!
memcmp
(
&
ext_header
.
os2
.
ne_
magic
,
"NE"
,
2
))
{
{
/* This is a Windows executable (NE) header. This can
/* This is a Windows executable (NE) header. This can
* mean either a 16-bit OS/2 or a 16-bit Windows or even a
* mean either a 16-bit OS/2 or a 16-bit Windows or even a
* DOS program (running under a DOS extender). To decide
* DOS program (running under a DOS extender). To decide
* which, we'll have to read the NE header.
* which, we'll have to read the NE header.
*/
*/
IMAGE_OS2_HEADER
ne
;
if
(
len
>=
sizeof
(
ext_header
.
os2
))
if
(
SetFilePointer
(
hfile
,
header
.
mz
.
e_lfanew
,
NULL
,
SEEK_SET
)
!=
-
1
&&
ReadFile
(
hfile
,
&
ne
,
sizeof
(
ne
),
&
len
,
NULL
)
&&
len
==
sizeof
(
ne
)
)
{
{
switch
(
ne
.
ne_exetyp
)
switch
(
ext_header
.
os2
.
ne_exetyp
)
{
{
case
2
:
return
BINARY_WIN16
;
case
2
:
return
BINARY_WIN16
;
case
5
:
return
BINARY_DOS
;
case
5
:
return
BINARY_DOS
;
default:
return
MODULE_Decide_OS2_OldWin
(
hfile
,
&
header
.
mz
,
&
ne
);
default:
return
MODULE_Decide_OS2_OldWin
(
hfile
,
&
header
.
mz
,
&
ext_header
.
os2
);
}
}
}
}
/* Couldn't read header, so abort. */
/* Couldn't read header, so abort. */
...
@@ -295,7 +300,7 @@ BOOL WINAPI GetBinaryTypeW( LPCWSTR lpApplicationName, LPDWORD lpBinaryType )
...
@@ -295,7 +300,7 @@ BOOL WINAPI GetBinaryTypeW( LPCWSTR lpApplicationName, LPDWORD lpBinaryType )
/* Check binary type
/* Check binary type
*/
*/
switch
(
MODULE_GetBinaryType
(
hfile
))
switch
(
MODULE_GetBinaryType
(
hfile
,
NULL
,
NULL
))
{
{
case
BINARY_UNKNOWN
:
case
BINARY_UNKNOWN
:
{
{
...
...
dlls/kernel/process.c
View file @
10e6f57f
...
@@ -389,6 +389,7 @@ static BOOL build_initial_environment( char **environ )
...
@@ -389,6 +389,7 @@ static BOOL build_initial_environment( char **environ )
if
(
!
strncmp
(
str
,
"WINE"
,
4
))
if
(
!
strncmp
(
str
,
"WINE"
,
4
))
{
{
if
(
is_special_env_var
(
str
+
4
))
str
+=
4
;
if
(
is_special_env_var
(
str
+
4
))
str
+=
4
;
else
if
(
!
strncmp
(
str
,
"WINEPRELOADRESERVE="
,
19
))
continue
;
/* skip it */
}
}
else
if
(
is_special_env_var
(
str
))
continue
;
/* skip it */
else
if
(
is_special_env_var
(
str
))
continue
;
/* skip it */
...
@@ -1051,7 +1052,7 @@ void __wine_kernel_init(void)
...
@@ -1051,7 +1052,7 @@ void __wine_kernel_init(void)
ExitProcess
(
1
);
ExitProcess
(
1
);
}
}
switch
(
MODULE_GetBinaryType
(
main_exe_file
))
switch
(
MODULE_GetBinaryType
(
main_exe_file
,
NULL
,
NULL
))
{
{
case
BINARY_PE_EXE
:
case
BINARY_PE_EXE
:
TRACE
(
"starting Win32 binary %s
\n
"
,
debugstr_w
(
main_exe_name
)
);
TRACE
(
"starting Win32 binary %s
\n
"
,
debugstr_w
(
main_exe_name
)
);
...
@@ -1257,26 +1258,28 @@ static char *alloc_env_string( const char *name, const char *value )
...
@@ -1257,26 +1258,28 @@ static char *alloc_env_string( const char *name, const char *value )
*
*
* Build the environment of a new child process.
* Build the environment of a new child process.
*/
*/
static
char
**
build_envp
(
const
WCHAR
*
envW
)
static
char
**
build_envp
(
const
WCHAR
*
envW
,
char
*
extra_env
)
{
{
const
WCHAR
*
p
;
const
WCHAR
*
end
;
char
**
envp
;
char
**
envp
;
char
*
env
;
char
*
env
,
*
p
;
int
count
=
0
,
length
;
int
count
=
0
,
length
;
for
(
p
=
envW
;
*
p
;
count
++
)
p
+=
strlenW
(
p
)
+
1
;
for
(
end
=
envW
;
*
end
;
count
++
)
end
+=
strlenW
(
end
)
+
1
;
p
++
;
end
++
;
length
=
WideCharToMultiByte
(
CP_UNIXCP
,
0
,
envW
,
p
-
envW
,
NULL
,
0
,
NULL
,
NULL
);
length
=
WideCharToMultiByte
(
CP_UNIXCP
,
0
,
envW
,
end
-
envW
,
NULL
,
0
,
NULL
,
NULL
);
if
(
!
(
env
=
malloc
(
length
)))
return
NULL
;
if
(
!
(
env
=
malloc
(
length
)))
return
NULL
;
WideCharToMultiByte
(
CP_UNIXCP
,
0
,
envW
,
p
-
envW
,
env
,
length
,
NULL
,
NULL
);
WideCharToMultiByte
(
CP_UNIXCP
,
0
,
envW
,
end
-
envW
,
env
,
length
,
NULL
,
NULL
);
if
(
extra_env
)
for
(
p
=
extra_env
;
*
p
;
p
+=
strlen
(
p
)
+
1
)
count
++
;
count
+=
4
;
count
+=
4
;
if
((
envp
=
malloc
(
count
*
sizeof
(
*
envp
)
)))
if
((
envp
=
malloc
(
count
*
sizeof
(
*
envp
)
)))
{
{
char
**
envptr
=
envp
;
char
**
envptr
=
envp
;
char
*
p
;
/* first the extra strings */
for
(
p
=
extra_env
;
*
p
;
p
+=
strlen
(
p
)
+
1
)
*
envptr
++
=
alloc_env_string
(
""
,
p
);
/* then put PATH, TEMP, TMP, HOME and WINEPREFIX from the unix env */
/* then put PATH, TEMP, TMP, HOME and WINEPREFIX from the unix env */
if
((
p
=
getenv
(
"PATH"
)))
*
envptr
++
=
alloc_env_string
(
"PATH="
,
p
);
if
((
p
=
getenv
(
"PATH"
)))
*
envptr
++
=
alloc_env_string
(
"PATH="
,
p
);
if
((
p
=
getenv
(
"TEMP"
)))
*
envptr
++
=
alloc_env_string
(
"TEMP="
,
p
);
if
((
p
=
getenv
(
"TEMP"
)))
*
envptr
++
=
alloc_env_string
(
"TEMP="
,
p
);
...
@@ -1289,6 +1292,7 @@ static char **build_envp( const WCHAR *envW )
...
@@ -1289,6 +1292,7 @@ static char **build_envp( const WCHAR *envW )
if
(
is_special_env_var
(
p
))
/* prefix it with "WINE" */
if
(
is_special_env_var
(
p
))
/* prefix it with "WINE" */
*
envptr
++
=
alloc_env_string
(
"WINE"
,
p
);
*
envptr
++
=
alloc_env_string
(
"WINE"
,
p
);
else
if
(
strncmp
(
p
,
"HOME="
,
5
)
&&
else
if
(
strncmp
(
p
,
"HOME="
,
5
)
&&
strncmp
(
p
,
"WINEPRELOADRESERVE="
,
19
)
&&
strncmp
(
p
,
"WINEPREFIX="
,
11
))
*
envptr
++
=
p
;
strncmp
(
p
,
"WINEPREFIX="
,
11
))
*
envptr
++
=
p
;
}
}
*
envptr
=
0
;
*
envptr
=
0
;
...
@@ -1319,7 +1323,7 @@ static int fork_and_exec( const char *filename, const WCHAR *cmdline,
...
@@ -1319,7 +1323,7 @@ static int fork_and_exec( const char *filename, const WCHAR *cmdline,
if
(
!
(
pid
=
fork
()))
/* child */
if
(
!
(
pid
=
fork
()))
/* child */
{
{
char
**
argv
=
build_argv
(
cmdline
,
0
);
char
**
argv
=
build_argv
(
cmdline
,
0
);
char
**
envp
=
build_envp
(
env
);
char
**
envp
=
build_envp
(
env
,
NULL
);
close
(
fd
[
0
]
);
close
(
fd
[
0
]
);
/* Reset signals that we previously set to SIG_IGN */
/* Reset signals that we previously set to SIG_IGN */
...
@@ -1405,7 +1409,8 @@ static RTL_USER_PROCESS_PARAMETERS *create_user_params( LPCWSTR filename, LPCWST
...
@@ -1405,7 +1409,8 @@ static RTL_USER_PROCESS_PARAMETERS *create_user_params( LPCWSTR filename, LPCWST
static
BOOL
create_process
(
HANDLE
hFile
,
LPCWSTR
filename
,
LPWSTR
cmd_line
,
LPWSTR
env
,
static
BOOL
create_process
(
HANDLE
hFile
,
LPCWSTR
filename
,
LPWSTR
cmd_line
,
LPWSTR
env
,
LPCWSTR
cur_dir
,
LPSECURITY_ATTRIBUTES
psa
,
LPSECURITY_ATTRIBUTES
tsa
,
LPCWSTR
cur_dir
,
LPSECURITY_ATTRIBUTES
psa
,
LPSECURITY_ATTRIBUTES
tsa
,
BOOL
inherit
,
DWORD
flags
,
LPSTARTUPINFOW
startup
,
BOOL
inherit
,
DWORD
flags
,
LPSTARTUPINFOW
startup
,
LPPROCESS_INFORMATION
info
,
LPCSTR
unixdir
)
LPPROCESS_INFORMATION
info
,
LPCSTR
unixdir
,
void
*
res_start
,
void
*
res_end
)
{
{
BOOL
ret
,
success
=
FALSE
;
BOOL
ret
,
success
=
FALSE
;
HANDLE
process_info
;
HANDLE
process_info
;
...
@@ -1415,12 +1420,16 @@ static BOOL create_process( HANDLE hFile, LPCWSTR filename, LPWSTR cmd_line, LPW
...
@@ -1415,12 +1420,16 @@ static BOOL create_process( HANDLE hFile, LPCWSTR filename, LPWSTR cmd_line, LPW
pid_t
pid
;
pid_t
pid
;
int
err
;
int
err
;
char
dummy
=
0
;
char
dummy
=
0
;
char
preloader_reserve
[
64
];
if
(
!
env
)
env
=
GetEnvironmentStringsW
();
if
(
!
env
)
env
=
GetEnvironmentStringsW
();
if
(
!
(
params
=
create_user_params
(
filename
,
cmd_line
,
cur_dir
,
startup
)))
if
(
!
(
params
=
create_user_params
(
filename
,
cmd_line
,
cur_dir
,
startup
)))
return
FALSE
;
return
FALSE
;
sprintf
(
preloader_reserve
,
"WINEPRELOADRESERVE=%lx-%lx%c"
,
(
unsigned
long
)
res_start
,
(
unsigned
long
)
res_end
,
0
);
/* create the synchronization pipes */
/* create the synchronization pipes */
if
(
pipe
(
startfd
)
==
-
1
)
if
(
pipe
(
startfd
)
==
-
1
)
...
@@ -1444,7 +1453,7 @@ static BOOL create_process( HANDLE hFile, LPCWSTR filename, LPWSTR cmd_line, LPW
...
@@ -1444,7 +1453,7 @@ static BOOL create_process( HANDLE hFile, LPCWSTR filename, LPWSTR cmd_line, LPW
if
(
!
(
pid
=
fork
()))
/* child */
if
(
!
(
pid
=
fork
()))
/* child */
{
{
char
**
argv
=
build_argv
(
cmd_line
,
1
);
char
**
argv
=
build_argv
(
cmd_line
,
1
);
char
**
envp
=
build_envp
(
env
);
char
**
envp
=
build_envp
(
env
,
preloader_reserve
);
close
(
startfd
[
1
]
);
close
(
startfd
[
1
]
);
close
(
execfd
[
0
]
);
close
(
execfd
[
0
]
);
...
@@ -1604,7 +1613,7 @@ static BOOL create_vdm_process( LPCWSTR filename, LPWSTR cmd_line, LPWSTR env, L
...
@@ -1604,7 +1613,7 @@ static BOOL create_vdm_process( LPCWSTR filename, LPWSTR cmd_line, LPWSTR env, L
}
}
sprintfW
(
new_cmd_line
,
argsW
,
winevdmW
,
filename
,
cmd_line
);
sprintfW
(
new_cmd_line
,
argsW
,
winevdmW
,
filename
,
cmd_line
);
ret
=
create_process
(
0
,
winevdmW
,
new_cmd_line
,
env
,
cur_dir
,
psa
,
tsa
,
inherit
,
ret
=
create_process
(
0
,
winevdmW
,
new_cmd_line
,
env
,
cur_dir
,
psa
,
tsa
,
inherit
,
flags
,
startup
,
info
,
unixdir
);
flags
,
startup
,
info
,
unixdir
,
NULL
,
NULL
);
HeapFree
(
GetProcessHeap
(),
0
,
new_cmd_line
);
HeapFree
(
GetProcessHeap
(),
0
,
new_cmd_line
);
return
ret
;
return
ret
;
}
}
...
@@ -1784,6 +1793,7 @@ BOOL WINAPI CreateProcessW( LPCWSTR app_name, LPWSTR cmd_line, LPSECURITY_ATTRIB
...
@@ -1784,6 +1793,7 @@ BOOL WINAPI CreateProcessW( LPCWSTR app_name, LPWSTR cmd_line, LPSECURITY_ATTRIB
char
*
unixdir
=
NULL
;
char
*
unixdir
=
NULL
;
WCHAR
name
[
MAX_PATH
];
WCHAR
name
[
MAX_PATH
];
WCHAR
*
tidy_cmdline
,
*
p
,
*
envW
=
env
;
WCHAR
*
tidy_cmdline
,
*
p
,
*
envW
=
env
;
void
*
res_start
,
*
res_end
;
/* Process the AppName and/or CmdLine to get module name and path */
/* Process the AppName and/or CmdLine to get module name and path */
...
@@ -1833,16 +1843,16 @@ BOOL WINAPI CreateProcessW( LPCWSTR app_name, LPWSTR cmd_line, LPSECURITY_ATTRIB
...
@@ -1833,16 +1843,16 @@ BOOL WINAPI CreateProcessW( LPCWSTR app_name, LPWSTR cmd_line, LPSECURITY_ATTRIB
{
{
TRACE
(
"starting %s as Winelib app
\n
"
,
debugstr_w
(
name
)
);
TRACE
(
"starting %s as Winelib app
\n
"
,
debugstr_w
(
name
)
);
retv
=
create_process
(
0
,
name
,
tidy_cmdline
,
envW
,
cur_dir
,
process_attr
,
thread_attr
,
retv
=
create_process
(
0
,
name
,
tidy_cmdline
,
envW
,
cur_dir
,
process_attr
,
thread_attr
,
inherit
,
flags
,
startup_info
,
info
,
unixdir
);
inherit
,
flags
,
startup_info
,
info
,
unixdir
,
NULL
,
NULL
);
goto
done
;
goto
done
;
}
}
switch
(
MODULE_GetBinaryType
(
hFile
))
switch
(
MODULE_GetBinaryType
(
hFile
,
&
res_start
,
&
res_end
))
{
{
case
BINARY_PE_EXE
:
case
BINARY_PE_EXE
:
TRACE
(
"starting %s as Win32 binary
\n
"
,
debugstr_w
(
name
)
);
TRACE
(
"starting %s as Win32 binary
(%p-%p)
\n
"
,
debugstr_w
(
name
),
res_start
,
res_end
);
retv
=
create_process
(
hFile
,
name
,
tidy_cmdline
,
envW
,
cur_dir
,
process_attr
,
thread_attr
,
retv
=
create_process
(
hFile
,
name
,
tidy_cmdline
,
envW
,
cur_dir
,
process_attr
,
thread_attr
,
inherit
,
flags
,
startup_info
,
info
,
unixdir
);
inherit
,
flags
,
startup_info
,
info
,
unixdir
,
res_start
,
res_end
);
break
;
break
;
case
BINARY_WIN16
:
case
BINARY_WIN16
:
case
BINARY_DOS
:
case
BINARY_DOS
:
...
@@ -1861,7 +1871,7 @@ BOOL WINAPI CreateProcessW( LPCWSTR app_name, LPWSTR cmd_line, LPSECURITY_ATTRIB
...
@@ -1861,7 +1871,7 @@ BOOL WINAPI CreateProcessW( LPCWSTR app_name, LPWSTR cmd_line, LPSECURITY_ATTRIB
case
BINARY_UNIX_LIB
:
case
BINARY_UNIX_LIB
:
TRACE
(
"%s is a Unix library, starting as Winelib app
\n
"
,
debugstr_w
(
name
)
);
TRACE
(
"%s is a Unix library, starting as Winelib app
\n
"
,
debugstr_w
(
name
)
);
retv
=
create_process
(
hFile
,
name
,
tidy_cmdline
,
envW
,
cur_dir
,
process_attr
,
thread_attr
,
retv
=
create_process
(
hFile
,
name
,
tidy_cmdline
,
envW
,
cur_dir
,
process_attr
,
thread_attr
,
inherit
,
flags
,
startup_info
,
info
,
unixdir
);
inherit
,
flags
,
startup_info
,
info
,
unixdir
,
NULL
,
NULL
);
break
;
break
;
case
BINARY_UNKNOWN
:
case
BINARY_UNKNOWN
:
/* check for .com or .bat extension */
/* check for .com or .bat extension */
...
...
include/module.h
View file @
10e6f57f
...
@@ -153,7 +153,7 @@ enum binary_type
...
@@ -153,7 +153,7 @@ enum binary_type
/* module.c */
/* module.c */
extern
NTSTATUS
MODULE_DllThreadAttach
(
LPVOID
lpReserved
);
extern
NTSTATUS
MODULE_DllThreadAttach
(
LPVOID
lpReserved
);
extern
enum
binary_type
MODULE_GetBinaryType
(
HANDLE
hfile
);
extern
enum
binary_type
MODULE_GetBinaryType
(
HANDLE
hfile
,
void
**
res_start
,
void
**
res_end
);
/* ne_module.c */
/* ne_module.c */
extern
NE_MODULE
*
NE_GetPtr
(
HMODULE16
hModule
);
extern
NE_MODULE
*
NE_GetPtr
(
HMODULE16
hModule
);
...
...
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