2012-04-30 13:55:29 +10:00
/*
* Copyright 2012 Red Hat Inc .
*
* Permission is hereby granted , free of charge , to any person obtaining a
* copy of this software and associated documentation files ( the " Software " ) ,
* to deal in the Software without restriction , including without limitation
* the rights to use , copy , modify , merge , publish , distribute , sublicense ,
* and / or sell copies of the Software , and to permit persons to whom the
* Software is furnished to do so , subject to the following conditions :
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
* IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL
* THE COPYRIGHT HOLDER ( S ) OR AUTHOR ( S ) BE LIABLE FOR ANY CLAIM , DAMAGES OR
* OTHER LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE ,
* ARISING FROM , OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE .
*
* Authors : Ben Skeggs
*/
2012-07-20 08:17:34 +10:00
# include "nouveau_drm.h"
2012-04-30 13:55:29 +10:00
# include "nouveau_dma.h"
# include "nouveau_fence.h"
struct nv04_fence_chan {
struct nouveau_fence_chan base ;
} ;
struct nv04_fence_priv {
struct nouveau_fence_priv base ;
} ;
static int
nv04_fence_emit ( struct nouveau_fence * fence )
{
struct nouveau_channel * chan = fence - > channel ;
int ret = RING_SPACE ( chan , 2 ) ;
if ( ret = = 0 ) {
BEGIN_NV04 ( chan , NvSubSw , 0x0150 , 1 ) ;
2014-01-09 11:03:11 +01:00
OUT_RING ( chan , fence - > base . seqno ) ;
2012-04-30 13:55:29 +10:00
FIRE_RING ( chan ) ;
}
return ret ;
}
static int
2012-05-04 16:25:47 +10:00
nv04_fence_sync ( struct nouveau_fence * fence ,
struct nouveau_channel * prev , struct nouveau_channel * chan )
2012-04-30 13:55:29 +10:00
{
return - ENODEV ;
}
static u32
nv04_fence_read ( struct nouveau_channel * chan )
{
2015-08-20 14:54:18 +10:00
struct nv04_nvsw_get_ref_v0 args = { } ;
WARN_ON ( nvif_object_mthd ( & chan - > nvsw , NV04_NVSW_GET_REF ,
& args , sizeof ( args ) ) ) ;
return args . ref ;
2012-04-30 13:55:29 +10:00
}
static void
2012-07-19 10:51:42 +10:00
nv04_fence_context_del ( struct nouveau_channel * chan )
2012-04-30 13:55:29 +10:00
{
2012-07-19 10:51:42 +10:00
struct nv04_fence_chan * fctx = chan - > fence ;
2012-04-30 13:55:29 +10:00
nouveau_fence_context_del ( & fctx - > base ) ;
2012-07-19 10:51:42 +10:00
chan - > fence = NULL ;
2014-09-29 10:06:18 +02:00
nouveau_fence_context_free ( & fctx - > base ) ;
2012-04-30 13:55:29 +10:00
}
static int
2012-07-19 10:51:42 +10:00
nv04_fence_context_new ( struct nouveau_channel * chan )
2012-04-30 13:55:29 +10:00
{
struct nv04_fence_chan * fctx = kzalloc ( sizeof ( * fctx ) , GFP_KERNEL ) ;
if ( fctx ) {
2014-01-09 11:03:11 +01:00
nouveau_fence_context_new ( chan , & fctx - > base ) ;
2013-02-14 13:20:17 +10:00
fctx - > base . emit = nv04_fence_emit ;
fctx - > base . sync = nv04_fence_sync ;
fctx - > base . read = nv04_fence_read ;
2012-07-19 10:51:42 +10:00
chan - > fence = fctx ;
2012-04-30 13:55:29 +10:00
return 0 ;
}
return - ENOMEM ;
}
static void
2012-07-20 08:17:34 +10:00
nv04_fence_destroy ( struct nouveau_drm * drm )
2012-04-30 13:55:29 +10:00
{
2012-07-20 08:17:34 +10:00
struct nv04_fence_priv * priv = drm - > fence ;
drm - > fence = NULL ;
2012-04-30 13:55:29 +10:00
kfree ( priv ) ;
}
int
2012-07-20 08:17:34 +10:00
nv04_fence_create ( struct nouveau_drm * drm )
2012-04-30 13:55:29 +10:00
{
struct nv04_fence_priv * priv ;
2012-07-20 08:17:34 +10:00
priv = drm - > fence = kzalloc ( sizeof ( * priv ) , GFP_KERNEL ) ;
2012-04-30 13:55:29 +10:00
if ( ! priv )
return - ENOMEM ;
2012-07-19 10:51:42 +10:00
priv - > base . dtor = nv04_fence_destroy ;
priv - > base . context_new = nv04_fence_context_new ;
priv - > base . context_del = nv04_fence_context_del ;
2014-01-09 11:03:11 +01:00
priv - > base . contexts = 15 ;
priv - > base . context_base = fence_context_alloc ( priv - > base . contexts ) ;
2012-07-20 08:17:34 +10:00
return 0 ;
2012-04-30 13:55:29 +10:00
}