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
53b3f9d4
Commit
53b3f9d4
authored
Apr 10, 2012
by
Henri Verbeet
Committed by
Alexandre Julliard
Apr 11, 2012
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
d3d9: COM cleanup for the IDirect3DSwapChain9 interface.
This is mostly based on the COM cleanup scripts by Michael Stefaniuc.
parent
8b95d8d2
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
58 additions
and
43 deletions
+58
-43
d3d9_private.h
dlls/d3d9/d3d9_private.h
+1
-1
device.c
dlls/d3d9/device.c
+2
-2
swapchain.c
dlls/d3d9/swapchain.c
+55
-40
No files found.
dlls/d3d9/d3d9_private.h
View file @
53b3f9d4
...
@@ -205,7 +205,7 @@ HRESULT volume_init(IDirect3DVolume9Impl *volume, IDirect3DDevice9Impl *device,
...
@@ -205,7 +205,7 @@ HRESULT volume_init(IDirect3DVolume9Impl *volume, IDirect3DDevice9Impl *device,
typedef
struct
IDirect3DSwapChain9Impl
typedef
struct
IDirect3DSwapChain9Impl
{
{
/* IUnknown fields */
/* IUnknown fields */
const
IDirect3DSwapChain9Vtbl
*
lpVtbl
;
IDirect3DSwapChain9
IDirect3DSwapChain9_iface
;
LONG
ref
;
LONG
ref
;
struct
wined3d_swapchain
*
wined3d_swapchain
;
struct
wined3d_swapchain
*
wined3d_swapchain
;
IDirect3DDevice9Ex
*
parentDevice
;
IDirect3DDevice9Ex
*
parentDevice
;
...
...
dlls/d3d9/device.c
View file @
53b3f9d4
...
@@ -451,7 +451,7 @@ static HRESULT WINAPI DECLSPEC_HOTPATCH IDirect3DDevice9Impl_CreateAdditionalSwa
...
@@ -451,7 +451,7 @@ static HRESULT WINAPI DECLSPEC_HOTPATCH IDirect3DDevice9Impl_CreateAdditionalSwa
iface
,
present_parameters
,
swapchain
);
iface
,
present_parameters
,
swapchain
);
if
(
SUCCEEDED
(
hr
=
d3d9_swapchain_create
(
device
,
present_parameters
,
&
object
)))
if
(
SUCCEEDED
(
hr
=
d3d9_swapchain_create
(
device
,
present_parameters
,
&
object
)))
*
swapchain
=
(
IDirect3DSwapChain9
*
)
object
;
*
swapchain
=
&
object
->
IDirect3DSwapChain9_iface
;
return
hr
;
return
hr
;
}
}
...
@@ -3307,7 +3307,7 @@ static HRESULT CDECL device_parent_create_swapchain(struct wined3d_device_parent
...
@@ -3307,7 +3307,7 @@ static HRESULT CDECL device_parent_create_swapchain(struct wined3d_device_parent
*
swapchain
=
d3d_swapchain
->
wined3d_swapchain
;
*
swapchain
=
d3d_swapchain
->
wined3d_swapchain
;
wined3d_swapchain_incref
(
*
swapchain
);
wined3d_swapchain_incref
(
*
swapchain
);
IDirect3DSwapChain9_Release
(
(
IDirect3DSwapChain9
*
)
d3d_swapchain
);
IDirect3DSwapChain9_Release
(
&
d3d_swapchain
->
IDirect3DSwapChain9_iface
);
/* Copy back the presentation parameters */
/* Copy back the presentation parameters */
desc
->
backbuffer_width
=
local_parameters
.
BackBufferWidth
;
desc
->
backbuffer_width
=
local_parameters
.
BackBufferWidth
;
...
...
dlls/d3d9/swapchain.c
View file @
53b3f9d4
...
@@ -25,55 +25,61 @@
...
@@ -25,55 +25,61 @@
WINE_DEFAULT_DEBUG_CHANNEL
(
d3d9
);
WINE_DEFAULT_DEBUG_CHANNEL
(
d3d9
);
/* IDirect3DSwapChain IUnknown parts follow: */
static
inline
IDirect3DSwapChain9Impl
*
impl_from_IDirect3DSwapChain9
(
IDirect3DSwapChain9
*
iface
)
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_QueryInterface
(
LPDIRECT3DSWAPCHAIN9
iface
,
REFIID
riid
,
LPVOID
*
ppobj
)
{
{
TRACE
(
"iface %p, riid %s, object %p.
\n
"
,
iface
,
debugstr_guid
(
riid
),
ppobj
);
return
CONTAINING_RECORD
(
iface
,
IDirect3DSwapChain9Impl
,
IDirect3DSwapChain9_iface
);
}
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_QueryInterface
(
IDirect3DSwapChain9
*
iface
,
REFIID
riid
,
void
**
out
)
{
TRACE
(
"iface %p, riid %s, out %p.
\n
"
,
iface
,
debugstr_guid
(
riid
),
out
);
if
(
IsEqualGUID
(
riid
,
&
IID_IDirect3DSwapChain9
)
if
(
IsEqualGUID
(
riid
,
&
IID_IDirect3DSwapChain9
)
||
IsEqualGUID
(
riid
,
&
IID_IUnknown
))
||
IsEqualGUID
(
riid
,
&
IID_IUnknown
))
{
{
IDirect3DSwapChain9_AddRef
(
iface
);
IDirect3DSwapChain9_AddRef
(
iface
);
*
ppobj
=
iface
;
*
out
=
iface
;
return
S_OK
;
return
S_OK
;
}
}
WARN
(
"%s not implemented, returning E_NOINTERFACE.
\n
"
,
debugstr_guid
(
riid
));
WARN
(
"%s not implemented, returning E_NOINTERFACE.
\n
"
,
debugstr_guid
(
riid
));
*
ppobj
=
NULL
;
*
out
=
NULL
;
return
E_NOINTERFACE
;
return
E_NOINTERFACE
;
}
}
static
ULONG
WINAPI
IDirect3DSwapChain9Impl_AddRef
(
LPDIRECT3DSWAPCHAIN9
iface
)
{
static
ULONG
WINAPI
IDirect3DSwapChain9Impl_AddRef
(
IDirect3DSwapChain9
*
iface
)
IDirect3DSwapChain9Impl
*
This
=
(
IDirect3DSwapChain9Impl
*
)
iface
;
{
ULONG
ref
=
InterlockedIncrement
(
&
This
->
ref
);
IDirect3DSwapChain9Impl
*
swapchain
=
impl_from_IDirect3DSwapChain9
(
iface
);
ULONG
ref
=
InterlockedIncrement
(
&
swapchain
->
ref
);
TRACE
(
"%p increasing refcount to %u.
\n
"
,
iface
,
ref
);
TRACE
(
"%p increasing refcount to %u.
\n
"
,
iface
,
ref
);
if
(
ref
==
1
)
if
(
ref
==
1
)
{
{
if
(
This
->
parentDevice
)
if
(
swapchain
->
parentDevice
)
IDirect3DDevice9Ex_AddRef
(
This
->
parentDevice
);
IDirect3DDevice9Ex_AddRef
(
swapchain
->
parentDevice
);
wined3d_mutex_lock
();
wined3d_mutex_lock
();
wined3d_swapchain_incref
(
This
->
wined3d_swapchain
);
wined3d_swapchain_incref
(
swapchain
->
wined3d_swapchain
);
wined3d_mutex_unlock
();
wined3d_mutex_unlock
();
}
}
return
ref
;
return
ref
;
}
}
static
ULONG
WINAPI
IDirect3DSwapChain9Impl_Release
(
LPDIRECT3DSWAPCHAIN9
iface
)
{
static
ULONG
WINAPI
IDirect3DSwapChain9Impl_Release
(
IDirect3DSwapChain9
*
iface
)
IDirect3DSwapChain9Impl
*
This
=
(
IDirect3DSwapChain9Impl
*
)
iface
;
{
ULONG
ref
=
InterlockedDecrement
(
&
This
->
ref
);
IDirect3DSwapChain9Impl
*
swapchain
=
impl_from_IDirect3DSwapChain9
(
iface
);
ULONG
ref
=
InterlockedDecrement
(
&
swapchain
->
ref
);
TRACE
(
"%p decreasing refcount to %u.
\n
"
,
iface
,
ref
);
TRACE
(
"%p decreasing refcount to %u.
\n
"
,
iface
,
ref
);
if
(
ref
==
0
)
{
if
(
ref
==
0
)
{
IDirect3DDevice9Ex
*
parentDevice
=
This
->
parentDevice
;
IDirect3DDevice9Ex
*
parentDevice
=
swapchain
->
parentDevice
;
wined3d_mutex_lock
();
wined3d_mutex_lock
();
wined3d_swapchain_decref
(
This
->
wined3d_swapchain
);
wined3d_swapchain_decref
(
swapchain
->
wined3d_swapchain
);
wined3d_mutex_unlock
();
wined3d_mutex_unlock
();
/* Release the device last, as it may cause the device to be destroyed. */
/* Release the device last, as it may cause the device to be destroyed. */
...
@@ -83,16 +89,20 @@ static ULONG WINAPI IDirect3DSwapChain9Impl_Release(LPDIRECT3DSWAPCHAIN9 iface)
...
@@ -83,16 +89,20 @@ static ULONG WINAPI IDirect3DSwapChain9Impl_Release(LPDIRECT3DSWAPCHAIN9 iface)
}
}
/* IDirect3DSwapChain9 parts follow: */
/* IDirect3DSwapChain9 parts follow: */
static
HRESULT
WINAPI
DECLSPEC_HOTPATCH
IDirect3DSwapChain9Impl_Present
(
LPDIRECT3DSWAPCHAIN9
iface
,
CONST
RECT
*
pSourceRect
,
CONST
RECT
*
pDestRect
,
HWND
hDestWindowOverride
,
CONST
RGNDATA
*
pDirtyRegion
,
DWORD
dwFlags
)
{
static
HRESULT
WINAPI
DECLSPEC_HOTPATCH
IDirect3DSwapChain9Impl_Present
(
IDirect3DSwapChain9
*
iface
,
IDirect3DSwapChain9Impl
*
This
=
(
IDirect3DSwapChain9Impl
*
)
iface
;
const
RECT
*
src_rect
,
const
RECT
*
dst_rect
,
HWND
dst_window_override
,
const
RGNDATA
*
dirty_region
,
DWORD
flags
)
{
IDirect3DSwapChain9Impl
*
swapchain
=
impl_from_IDirect3DSwapChain9
(
iface
);
HRESULT
hr
;
HRESULT
hr
;
TRACE
(
"iface %p, src_rect %p, dst_rect %p, dst_window_override %p, dirty_region %p, flags %#x.
\n
"
,
TRACE
(
"iface %p, src_rect %s, dst_rect %s, dst_window_override %p, dirty_region %p, flags %#x.
\n
"
,
iface
,
pSourceRect
,
pDestRect
,
hDestWindowOverride
,
pDirtyRegion
,
dwFlags
);
iface
,
wine_dbgstr_rect
(
src_rect
),
wine_dbgstr_rect
(
dst_rect
),
dst_window_override
,
dirty_region
,
flags
);
wined3d_mutex_lock
();
wined3d_mutex_lock
();
hr
=
wined3d_swapchain_present
(
This
->
wined3d_swapchain
,
pSourceR
ect
,
hr
=
wined3d_swapchain_present
(
swapchain
->
wined3d_swapchain
,
src_r
ect
,
pDestRect
,
hDestWindowOverride
,
pDirtyRegion
,
dwF
lags
);
dst_rect
,
dst_window_override
,
dirty_region
,
f
lags
);
wined3d_mutex_unlock
();
wined3d_mutex_unlock
();
return
hr
;
return
hr
;
...
@@ -101,14 +111,14 @@ static HRESULT WINAPI DECLSPEC_HOTPATCH IDirect3DSwapChain9Impl_Present(LPDIRECT
...
@@ -101,14 +111,14 @@ static HRESULT WINAPI DECLSPEC_HOTPATCH IDirect3DSwapChain9Impl_Present(LPDIRECT
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetFrontBufferData
(
IDirect3DSwapChain9
*
iface
,
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetFrontBufferData
(
IDirect3DSwapChain9
*
iface
,
IDirect3DSurface9
*
pDestSurface
)
IDirect3DSurface9
*
pDestSurface
)
{
{
IDirect3DSwapChain9Impl
*
This
=
(
IDirect3DSwapChain9Impl
*
)
iface
;
IDirect3DSwapChain9Impl
*
swapchain
=
impl_from_IDirect3DSwapChain9
(
iface
)
;
IDirect3DSurface9Impl
*
dst
=
unsafe_impl_from_IDirect3DSurface9
(
pDestSurface
);
IDirect3DSurface9Impl
*
dst
=
unsafe_impl_from_IDirect3DSurface9
(
pDestSurface
);
HRESULT
hr
;
HRESULT
hr
;
TRACE
(
"iface %p, surface %p.
\n
"
,
iface
,
pDestSurface
);
TRACE
(
"iface %p, surface %p.
\n
"
,
iface
,
pDestSurface
);
wined3d_mutex_lock
();
wined3d_mutex_lock
();
hr
=
wined3d_swapchain_get_front_buffer_data
(
This
->
wined3d_swapchain
,
dst
->
wined3d_surface
);
hr
=
wined3d_swapchain_get_front_buffer_data
(
swapchain
->
wined3d_swapchain
,
dst
->
wined3d_surface
);
wined3d_mutex_unlock
();
wined3d_mutex_unlock
();
return
hr
;
return
hr
;
...
@@ -117,7 +127,7 @@ static HRESULT WINAPI IDirect3DSwapChain9Impl_GetFrontBufferData(IDirect3DSwapCh
...
@@ -117,7 +127,7 @@ static HRESULT WINAPI IDirect3DSwapChain9Impl_GetFrontBufferData(IDirect3DSwapCh
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetBackBuffer
(
IDirect3DSwapChain9
*
iface
,
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetBackBuffer
(
IDirect3DSwapChain9
*
iface
,
UINT
iBackBuffer
,
D3DBACKBUFFER_TYPE
Type
,
IDirect3DSurface9
**
ppBackBuffer
)
UINT
iBackBuffer
,
D3DBACKBUFFER_TYPE
Type
,
IDirect3DSurface9
**
ppBackBuffer
)
{
{
IDirect3DSwapChain9Impl
*
This
=
(
IDirect3DSwapChain9Impl
*
)
iface
;
IDirect3DSwapChain9Impl
*
swapchain
=
impl_from_IDirect3DSwapChain9
(
iface
)
;
struct
wined3d_surface
*
wined3d_surface
=
NULL
;
struct
wined3d_surface
*
wined3d_surface
=
NULL
;
HRESULT
hr
;
HRESULT
hr
;
...
@@ -125,7 +135,7 @@ static HRESULT WINAPI IDirect3DSwapChain9Impl_GetBackBuffer(IDirect3DSwapChain9
...
@@ -125,7 +135,7 @@ static HRESULT WINAPI IDirect3DSwapChain9Impl_GetBackBuffer(IDirect3DSwapChain9
iface
,
iBackBuffer
,
Type
,
ppBackBuffer
);
iface
,
iBackBuffer
,
Type
,
ppBackBuffer
);
wined3d_mutex_lock
();
wined3d_mutex_lock
();
hr
=
wined3d_swapchain_get_back_buffer
(
This
->
wined3d_swapchain
,
hr
=
wined3d_swapchain_get_back_buffer
(
swapchain
->
wined3d_swapchain
,
iBackBuffer
,
(
enum
wined3d_backbuffer_type
)
Type
,
&
wined3d_surface
);
iBackBuffer
,
(
enum
wined3d_backbuffer_type
)
Type
,
&
wined3d_surface
);
if
(
SUCCEEDED
(
hr
)
&&
wined3d_surface
)
if
(
SUCCEEDED
(
hr
)
&&
wined3d_surface
)
{
{
...
@@ -139,41 +149,46 @@ static HRESULT WINAPI IDirect3DSwapChain9Impl_GetBackBuffer(IDirect3DSwapChain9
...
@@ -139,41 +149,46 @@ static HRESULT WINAPI IDirect3DSwapChain9Impl_GetBackBuffer(IDirect3DSwapChain9
return
hr
;
return
hr
;
}
}
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetRasterStatus
(
LPDIRECT3DSWAPCHAIN9
iface
,
D3DRASTER_STATUS
*
pRasterStatus
)
{
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetRasterStatus
(
IDirect3DSwapChain9
*
iface
,
IDirect3DSwapChain9Impl
*
This
=
(
IDirect3DSwapChain9Impl
*
)
iface
;
D3DRASTER_STATUS
*
raster_status
)
{
IDirect3DSwapChain9Impl
*
swapchain
=
impl_from_IDirect3DSwapChain9
(
iface
);
HRESULT
hr
;
HRESULT
hr
;
TRACE
(
"iface %p, raster_status %p.
\n
"
,
iface
,
pRasterS
tatus
);
TRACE
(
"iface %p, raster_status %p.
\n
"
,
iface
,
raster_s
tatus
);
wined3d_mutex_lock
();
wined3d_mutex_lock
();
hr
=
wined3d_swapchain_get_raster_status
(
This
->
wined3d_swapchain
,
(
struct
wined3d_raster_status
*
)
pRasterStatus
);
hr
=
wined3d_swapchain_get_raster_status
(
swapchain
->
wined3d_swapchain
,
(
struct
wined3d_raster_status
*
)
raster_status
);
wined3d_mutex_unlock
();
wined3d_mutex_unlock
();
return
hr
;
return
hr
;
}
}
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetDisplayMode
(
LPDIRECT3DSWAPCHAIN9
iface
,
D3DDISPLAYMODE
*
pMode
)
{
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetDisplayMode
(
IDirect3DSwapChain9
*
iface
,
D3DDISPLAYMODE
*
mode
)
IDirect3DSwapChain9Impl
*
This
=
(
IDirect3DSwapChain9Impl
*
)
iface
;
{
IDirect3DSwapChain9Impl
*
swapchain
=
impl_from_IDirect3DSwapChain9
(
iface
);
HRESULT
hr
;
HRESULT
hr
;
TRACE
(
"iface %p, mode %p.
\n
"
,
iface
,
pM
ode
);
TRACE
(
"iface %p, mode %p.
\n
"
,
iface
,
m
ode
);
wined3d_mutex_lock
();
wined3d_mutex_lock
();
hr
=
wined3d_swapchain_get_display_mode
(
This
->
wined3d_swapchain
,
(
struct
wined3d_display_mode
*
)
pM
ode
);
hr
=
wined3d_swapchain_get_display_mode
(
swapchain
->
wined3d_swapchain
,
(
struct
wined3d_display_mode
*
)
m
ode
);
wined3d_mutex_unlock
();
wined3d_mutex_unlock
();
if
(
SUCCEEDED
(
hr
))
pMode
->
Format
=
d3dformat_from_wined3dformat
(
pMode
->
Format
);
if
(
SUCCEEDED
(
hr
))
mode
->
Format
=
d3dformat_from_wined3dformat
(
mode
->
Format
);
return
hr
;
return
hr
;
}
}
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetDevice
(
IDirect3DSwapChain9
*
iface
,
IDirect3DDevice9
**
device
)
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetDevice
(
IDirect3DSwapChain9
*
iface
,
IDirect3DDevice9
**
device
)
{
{
IDirect3DSwapChain9Impl
*
This
=
(
IDirect3DSwapChain9Impl
*
)
iface
;
IDirect3DSwapChain9Impl
*
swapchain
=
impl_from_IDirect3DSwapChain9
(
iface
)
;
TRACE
(
"iface %p, device %p.
\n
"
,
iface
,
device
);
TRACE
(
"iface %p, device %p.
\n
"
,
iface
,
device
);
*
device
=
(
IDirect3DDevice9
*
)
This
->
parentDevice
;
*
device
=
(
IDirect3DDevice9
*
)
swapchain
->
parentDevice
;
IDirect3DDevice9_AddRef
(
*
device
);
IDirect3DDevice9_AddRef
(
*
device
);
TRACE
(
"Returning device %p.
\n
"
,
*
device
);
TRACE
(
"Returning device %p.
\n
"
,
*
device
);
...
@@ -184,14 +199,14 @@ static HRESULT WINAPI IDirect3DSwapChain9Impl_GetDevice(IDirect3DSwapChain9 *ifa
...
@@ -184,14 +199,14 @@ static HRESULT WINAPI IDirect3DSwapChain9Impl_GetDevice(IDirect3DSwapChain9 *ifa
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetPresentParameters
(
IDirect3DSwapChain9
*
iface
,
static
HRESULT
WINAPI
IDirect3DSwapChain9Impl_GetPresentParameters
(
IDirect3DSwapChain9
*
iface
,
D3DPRESENT_PARAMETERS
*
pPresentationParameters
)
D3DPRESENT_PARAMETERS
*
pPresentationParameters
)
{
{
IDirect3DSwapChain9Impl
*
This
=
(
IDirect3DSwapChain9Impl
*
)
iface
;
IDirect3DSwapChain9Impl
*
swapchain
=
impl_from_IDirect3DSwapChain9
(
iface
)
;
struct
wined3d_swapchain_desc
desc
;
struct
wined3d_swapchain_desc
desc
;
HRESULT
hr
;
HRESULT
hr
;
TRACE
(
"iface %p, parameters %p.
\n
"
,
iface
,
pPresentationParameters
);
TRACE
(
"iface %p, parameters %p.
\n
"
,
iface
,
pPresentationParameters
);
wined3d_mutex_lock
();
wined3d_mutex_lock
();
hr
=
wined3d_swapchain_get_desc
(
This
->
wined3d_swapchain
,
&
desc
);
hr
=
wined3d_swapchain_get_desc
(
swapchain
->
wined3d_swapchain
,
&
desc
);
wined3d_mutex_unlock
();
wined3d_mutex_unlock
();
pPresentationParameters
->
BackBufferWidth
=
desc
.
backbuffer_width
;
pPresentationParameters
->
BackBufferWidth
=
desc
.
backbuffer_width
;
...
@@ -244,7 +259,7 @@ static HRESULT swapchain_init(IDirect3DSwapChain9Impl *swapchain, IDirect3DDevic
...
@@ -244,7 +259,7 @@ static HRESULT swapchain_init(IDirect3DSwapChain9Impl *swapchain, IDirect3DDevic
HRESULT
hr
;
HRESULT
hr
;
swapchain
->
ref
=
1
;
swapchain
->
ref
=
1
;
swapchain
->
lpVtbl
=
&
Direct3DSwapChain9_Vtbl
;
swapchain
->
IDirect3DSwapChain9_iface
.
lpVtbl
=
&
Direct3DSwapChain9_Vtbl
;
desc
.
backbuffer_width
=
present_parameters
->
BackBufferWidth
;
desc
.
backbuffer_width
=
present_parameters
->
BackBufferWidth
;
desc
.
backbuffer_height
=
present_parameters
->
BackBufferHeight
;
desc
.
backbuffer_height
=
present_parameters
->
BackBufferHeight
;
...
...
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