2010-05-21 13:26:39 -07:00
/*
* Copyright © 2008 - 2010 Intel Corporation
*
* 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 ( including the next
* paragraph ) 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 AUTHORS OR COPYRIGHT HOLDERS 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 :
* Eric Anholt < eric @ anholt . net >
* Zou Nan hai < nanhai . zou @ intel . com >
* Xiang Hai hao < haihao . xiang @ intel . com >
*
*/
# include "drmP.h"
# include "drm.h"
# include "i915_drv.h"
2010-05-21 09:08:55 +08:00
# include "i915_drm.h"
2010-05-21 13:26:39 -07:00
# include "i915_trace.h"
2010-09-19 14:40:43 +01:00
# include "intel_drv.h"
2010-05-21 13:26:39 -07:00
2011-10-16 10:23:31 +02:00
/*
* 965 + support PIPE_CONTROL commands , which provide finer grained control
* over cache flushing .
*/
struct pipe_control {
struct drm_i915_gem_object * obj ;
volatile u32 * cpu_page ;
u32 gtt_offset ;
} ;
2011-01-20 17:00:10 +00:00
static inline int ring_space ( struct intel_ring_buffer * ring )
{
int space = ( ring - > head & HEAD_ADDR ) - ( ring - > tail + 8 ) ;
if ( space < 0 )
space + = ring - > size ;
return space ;
}
2011-01-04 17:34:02 +00:00
static int
2010-10-27 12:18:21 +01:00
render_ring_flush ( struct intel_ring_buffer * ring ,
2010-09-19 17:53:44 +01:00
u32 invalidate_domains ,
u32 flush_domains )
2010-05-21 13:26:39 -07:00
{
2010-10-27 12:18:21 +01:00
struct drm_device * dev = ring - > dev ;
2010-08-07 11:01:22 +01:00
u32 cmd ;
2011-01-04 17:34:02 +00:00
int ret ;
2010-08-07 11:01:22 +01:00
2011-03-19 22:26:49 +00:00
/*
* read / write caches :
*
* I915_GEM_DOMAIN_RENDER is always invalidated , but is
* only flushed if MI_NO_WRITE_FLUSH is unset . On 965 , it is
* also flushed at 2 d versus 3 d pipeline switches .
*
* read - only caches :
*
* I915_GEM_DOMAIN_SAMPLER is flushed on pre - 965 if
* MI_READ_FLUSH is set , and is always flushed on 965.
*
* I915_GEM_DOMAIN_COMMAND may not exist ?
*
* I915_GEM_DOMAIN_INSTRUCTION , which exists on 965 , is
* invalidated when MI_EXE_FLUSH is set .
*
* I915_GEM_DOMAIN_VERTEX , which exists on 965 , is
* invalidated with every MI_FLUSH .
*
* TLBs :
*
* On 965 , TLBs associated with I915_GEM_DOMAIN_COMMAND
* and I915_GEM_DOMAIN_CPU in are invalidated at PTE write and
* I915_GEM_DOMAIN_RENDER and I915_GEM_DOMAIN_SAMPLER
* are flushed at any MI_FLUSH .
*/
cmd = MI_FLUSH | MI_NO_WRITE_FLUSH ;
if ( ( invalidate_domains | flush_domains ) &
I915_GEM_DOMAIN_RENDER )
cmd & = ~ MI_NO_WRITE_FLUSH ;
if ( INTEL_INFO ( dev ) - > gen < 4 ) {
2010-05-21 13:26:39 -07:00
/*
2011-03-19 22:26:49 +00:00
* On the 965 , the sampler cache always gets flushed
* and this bit is reserved .
2010-05-21 13:26:39 -07:00
*/
2011-03-19 22:26:49 +00:00
if ( invalidate_domains & I915_GEM_DOMAIN_SAMPLER )
cmd | = MI_READ_FLUSH ;
}
if ( invalidate_domains & I915_GEM_DOMAIN_INSTRUCTION )
cmd | = MI_EXE_FLUSH ;
2010-05-21 13:26:39 -07:00
2011-03-19 22:26:49 +00:00
if ( invalidate_domains & I915_GEM_DOMAIN_COMMAND & &
( IS_G4X ( dev ) | | IS_GEN5 ( dev ) ) )
cmd | = MI_INVALIDATE_ISP ;
2010-11-30 14:07:47 +00:00
2011-03-19 22:26:49 +00:00
ret = intel_ring_begin ( ring , 2 ) ;
if ( ret )
return ret ;
2011-01-04 17:34:02 +00:00
2011-03-19 22:26:49 +00:00
intel_ring_emit ( ring , cmd ) ;
intel_ring_emit ( ring , MI_NOOP ) ;
intel_ring_advance ( ring ) ;
2011-01-04 17:34:02 +00:00
return 0 ;
2010-05-21 09:08:55 +08:00
}
2011-10-16 10:23:31 +02:00
/**
* Emits a PIPE_CONTROL with a non - zero post - sync operation , for
* implementing two workarounds on gen6 . From section 1.4 .7 .1
* " PIPE_CONTROL " of the Sandy Bridge PRM volume 2 part 1 :
*
* [ DevSNB - C + { W / A } ] Before any depth stall flush ( including those
* produced by non - pipelined state commands ) , software needs to first
* send a PIPE_CONTROL with no bits set except Post - Sync Operation ! =
* 0.
*
* [ Dev - SNB { W / A } ] : Before a PIPE_CONTROL with Write Cache Flush Enable
* = 1 , a PIPE_CONTROL with any non - zero post - sync - op is required .
*
* And the workaround for these two requires this workaround first :
*
* [ Dev - SNB { W / A } ] : Pipe - control with CS - stall bit set must be sent
* BEFORE the pipe - control with a post - sync op and no write - cache
* flushes .
*
* And this last workaround is tricky because of the requirements on
* that bit . From section 1.4 .7 .2 .3 " Stall " of the Sandy Bridge PRM
* volume 2 part 1 :
*
* " 1 of the following must also be set:
* - Render Target Cache Flush Enable ( [ 12 ] of DW1 )
* - Depth Cache Flush Enable ( [ 0 ] of DW1 )
* - Stall at Pixel Scoreboard ( [ 1 ] of DW1 )
* - Depth Stall ( [ 13 ] of DW1 )
* - Post - Sync Operation ( [ 13 ] of DW1 )
* - Notify Enable ( [ 8 ] of DW1 ) "
*
* The cache flushes require the workaround flush that triggered this
* one , so we can ' t use it . Depth stall would trigger the same .
* Post - sync nonzero is what triggered this second workaround , so we
* can ' t use that one either . Notify enable is IRQs , which aren ' t
* really our business . That leaves only stall at scoreboard .
*/
static int
intel_emit_post_sync_nonzero_flush ( struct intel_ring_buffer * ring )
{
struct pipe_control * pc = ring - > private ;
u32 scratch_addr = pc - > gtt_offset + 128 ;
int ret ;
ret = intel_ring_begin ( ring , 6 ) ;
if ( ret )
return ret ;
intel_ring_emit ( ring , GFX_OP_PIPE_CONTROL ( 5 ) ) ;
intel_ring_emit ( ring , PIPE_CONTROL_CS_STALL |
PIPE_CONTROL_STALL_AT_SCOREBOARD ) ;
intel_ring_emit ( ring , scratch_addr | PIPE_CONTROL_GLOBAL_GTT ) ; /* address */
intel_ring_emit ( ring , 0 ) ; /* low dword */
intel_ring_emit ( ring , 0 ) ; /* high dword */
intel_ring_emit ( ring , MI_NOOP ) ;
intel_ring_advance ( ring ) ;
ret = intel_ring_begin ( ring , 6 ) ;
if ( ret )
return ret ;
intel_ring_emit ( ring , GFX_OP_PIPE_CONTROL ( 5 ) ) ;
intel_ring_emit ( ring , PIPE_CONTROL_QW_WRITE ) ;
intel_ring_emit ( ring , scratch_addr | PIPE_CONTROL_GLOBAL_GTT ) ; /* address */
intel_ring_emit ( ring , 0 ) ;
intel_ring_emit ( ring , 0 ) ;
intel_ring_emit ( ring , MI_NOOP ) ;
intel_ring_advance ( ring ) ;
return 0 ;
}
static int
gen6_render_ring_flush ( struct intel_ring_buffer * ring ,
u32 invalidate_domains , u32 flush_domains )
{
u32 flags = 0 ;
struct pipe_control * pc = ring - > private ;
u32 scratch_addr = pc - > gtt_offset + 128 ;
int ret ;
/* Force SNB workarounds for PIPE_CONTROL flushes */
intel_emit_post_sync_nonzero_flush ( ring ) ;
/* Just flush everything. Experiments have shown that reducing the
* number of bits based on the write domains has little performance
* impact .
*/
flags | = PIPE_CONTROL_RENDER_TARGET_CACHE_FLUSH ;
flags | = PIPE_CONTROL_INSTRUCTION_CACHE_INVALIDATE ;
flags | = PIPE_CONTROL_TEXTURE_CACHE_INVALIDATE ;
flags | = PIPE_CONTROL_DEPTH_CACHE_FLUSH ;
flags | = PIPE_CONTROL_VF_CACHE_INVALIDATE ;
flags | = PIPE_CONTROL_CONST_CACHE_INVALIDATE ;
flags | = PIPE_CONTROL_STATE_CACHE_INVALIDATE ;
ret = intel_ring_begin ( ring , 6 ) ;
if ( ret )
return ret ;
intel_ring_emit ( ring , GFX_OP_PIPE_CONTROL ( 5 ) ) ;
intel_ring_emit ( ring , flags ) ;
intel_ring_emit ( ring , scratch_addr | PIPE_CONTROL_GLOBAL_GTT ) ;
intel_ring_emit ( ring , 0 ) ; /* lower dword */
intel_ring_emit ( ring , 0 ) ; /* uppwer dword */
intel_ring_emit ( ring , MI_NOOP ) ;
intel_ring_advance ( ring ) ;
return 0 ;
}
2010-10-27 12:18:21 +01:00
static void ring_write_tail ( struct intel_ring_buffer * ring ,
2010-10-22 17:02:41 +01:00
u32 value )
2010-09-16 10:43:12 +08:00
{
2010-10-27 12:18:21 +01:00
drm_i915_private_t * dev_priv = ring - > dev - > dev_private ;
2010-10-22 17:02:41 +01:00
I915_WRITE_TAIL ( ring , value ) ;
2010-09-16 10:43:12 +08:00
}
2010-10-27 12:18:21 +01:00
u32 intel_ring_get_active_head ( struct intel_ring_buffer * ring )
2010-05-21 09:08:55 +08:00
{
2010-10-27 12:18:21 +01:00
drm_i915_private_t * dev_priv = ring - > dev - > dev_private ;
u32 acthd_reg = INTEL_INFO ( ring - > dev ) - > gen > = 4 ?
2010-09-24 21:14:22 +02:00
RING_ACTHD ( ring - > mmio_base ) : ACTHD ;
2010-05-21 09:08:55 +08:00
return I915_READ ( acthd_reg ) ;
}
2010-10-27 12:18:21 +01:00
static int init_ring_common ( struct intel_ring_buffer * ring )
2010-05-21 09:08:55 +08:00
{
2010-10-27 12:18:21 +01:00
drm_i915_private_t * dev_priv = ring - > dev - > dev_private ;
2010-11-08 19:18:58 +00:00
struct drm_i915_gem_object * obj = ring - > obj ;
2010-05-21 09:08:55 +08:00
u32 head ;
/* Stop the ring if it's running. */
2010-08-02 17:06:59 +02:00
I915_WRITE_CTL ( ring , 0 ) ;
2010-08-02 17:06:23 +02:00
I915_WRITE_HEAD ( ring , 0 ) ;
2010-10-27 12:18:21 +01:00
ring - > write_tail ( ring , 0 ) ;
2010-05-21 09:08:55 +08:00
/* Initialize the ring. */
2010-11-08 19:18:58 +00:00
I915_WRITE_START ( ring , obj - > gtt_offset ) ;
2010-08-02 17:06:23 +02:00
head = I915_READ_HEAD ( ring ) & HEAD_ADDR ;
2010-05-21 09:08:55 +08:00
/* G45 ring initialization fails to reset head to zero */
if ( head ! = 0 ) {
2010-12-05 20:42:33 +00:00
DRM_DEBUG_KMS ( " %s head not reset to zero "
" ctl %08x head %08x tail %08x start %08x \n " ,
ring - > name ,
I915_READ_CTL ( ring ) ,
I915_READ_HEAD ( ring ) ,
I915_READ_TAIL ( ring ) ,
I915_READ_START ( ring ) ) ;
2010-05-21 09:08:55 +08:00
2010-08-02 17:06:23 +02:00
I915_WRITE_HEAD ( ring , 0 ) ;
2010-05-21 09:08:55 +08:00
2010-12-05 20:42:33 +00:00
if ( I915_READ_HEAD ( ring ) & HEAD_ADDR ) {
DRM_ERROR ( " failed to set %s head to zero "
" ctl %08x head %08x tail %08x start %08x \n " ,
ring - > name ,
I915_READ_CTL ( ring ) ,
I915_READ_HEAD ( ring ) ,
I915_READ_TAIL ( ring ) ,
I915_READ_START ( ring ) ) ;
}
2010-05-21 09:08:55 +08:00
}
2010-08-02 17:06:59 +02:00
I915_WRITE_CTL ( ring ,
2010-11-07 11:45:52 +00:00
( ( ring - > size - PAGE_SIZE ) & RING_NR_PAGES )
2010-10-29 21:44:37 +01:00
| RING_REPORT_64K | RING_VALID ) ;
2010-05-21 09:08:55 +08:00
/* If the head is still not zero, the ring is dead */
2010-10-28 11:18:07 +01:00
if ( ( I915_READ_CTL ( ring ) & RING_VALID ) = = 0 | |
2010-11-08 19:18:58 +00:00
I915_READ_START ( ring ) ! = obj - > gtt_offset | |
2010-10-28 11:18:07 +01:00
( I915_READ_HEAD ( ring ) & HEAD_ADDR ) ! = 0 ) {
2010-11-09 10:16:56 +00:00
DRM_ERROR ( " %s initialization failed "
" ctl %08x head %08x tail %08x start %08x \n " ,
ring - > name ,
I915_READ_CTL ( ring ) ,
I915_READ_HEAD ( ring ) ,
I915_READ_TAIL ( ring ) ,
I915_READ_START ( ring ) ) ;
return - EIO ;
2010-05-21 09:08:55 +08:00
}
2010-10-27 12:18:21 +01:00
if ( ! drm_core_check_feature ( ring - > dev , DRIVER_MODESET ) )
i915_kernel_lost_context ( ring - > dev ) ;
2010-05-21 09:08:55 +08:00
else {
2011-01-20 17:00:10 +00:00
ring - > head = I915_READ_HEAD ( ring ) ;
2010-08-02 16:29:44 +02:00
ring - > tail = I915_READ_TAIL ( ring ) & TAIL_ADDR ;
2011-01-20 17:00:10 +00:00
ring - > space = ring_space ( ring ) ;
2010-05-21 09:08:55 +08:00
}
2010-12-04 11:30:53 +00:00
2010-05-21 09:08:55 +08:00
return 0 ;
}
2010-12-15 09:56:50 +00:00
static int
init_pipe_control ( struct intel_ring_buffer * ring )
{
struct pipe_control * pc ;
struct drm_i915_gem_object * obj ;
int ret ;
if ( ring - > private )
return 0 ;
pc = kmalloc ( sizeof ( * pc ) , GFP_KERNEL ) ;
if ( ! pc )
return - ENOMEM ;
obj = i915_gem_alloc_object ( ring - > dev , 4096 ) ;
if ( obj = = NULL ) {
DRM_ERROR ( " Failed to allocate seqno page \n " ) ;
ret = - ENOMEM ;
goto err ;
}
2011-04-04 09:44:39 +01:00
i915_gem_object_set_cache_level ( obj , I915_CACHE_LLC ) ;
2010-12-15 09:56:50 +00:00
ret = i915_gem_object_pin ( obj , 4096 , true ) ;
if ( ret )
goto err_unref ;
pc - > gtt_offset = obj - > gtt_offset ;
pc - > cpu_page = kmap ( obj - > pages [ 0 ] ) ;
if ( pc - > cpu_page = = NULL )
goto err_unpin ;
pc - > obj = obj ;
ring - > private = pc ;
return 0 ;
err_unpin :
i915_gem_object_unpin ( obj ) ;
err_unref :
drm_gem_object_unreference ( & obj - > base ) ;
err :
kfree ( pc ) ;
return ret ;
}
static void
cleanup_pipe_control ( struct intel_ring_buffer * ring )
{
struct pipe_control * pc = ring - > private ;
struct drm_i915_gem_object * obj ;
if ( ! ring - > private )
return ;
obj = pc - > obj ;
kunmap ( obj - > pages [ 0 ] ) ;
i915_gem_object_unpin ( obj ) ;
drm_gem_object_unreference ( & obj - > base ) ;
kfree ( pc ) ;
ring - > private = NULL ;
}
2010-10-27 12:18:21 +01:00
static int init_render_ring ( struct intel_ring_buffer * ring )
2010-05-21 09:08:55 +08:00
{
2010-10-27 12:18:21 +01:00
struct drm_device * dev = ring - > dev ;
2010-12-04 11:30:53 +00:00
struct drm_i915_private * dev_priv = dev - > dev_private ;
2010-10-27 12:18:21 +01:00
int ret = init_ring_common ( ring ) ;
2010-08-30 16:12:42 +08:00
2010-09-17 00:32:17 +01:00
if ( INTEL_INFO ( dev ) - > gen > 3 ) {
2010-10-27 12:18:21 +01:00
int mode = VS_TIMER_DISPATCH < < 16 | VS_TIMER_DISPATCH ;
2010-08-30 16:12:42 +08:00
I915_WRITE ( MI_MODE , mode ) ;
2011-08-12 15:28:32 -07:00
if ( IS_GEN7 ( dev ) )
I915_WRITE ( GFX_MODE_GEN7 ,
GFX_MODE_DISABLE ( GFX_TLB_INVALIDATE_ALWAYS ) |
GFX_MODE_ENABLE ( GFX_REPLAY_MODE ) ) ;
2010-05-21 09:08:55 +08:00
}
2010-10-27 12:18:21 +01:00
2011-10-16 10:23:31 +02:00
if ( INTEL_INFO ( dev ) - > gen > = 5 ) {
2010-12-15 09:56:50 +00:00
ret = init_pipe_control ( ring ) ;
if ( ret )
return ret ;
}
2011-12-12 19:21:58 -08:00
if ( INTEL_INFO ( dev ) - > gen > = 6 ) {
I915_WRITE ( INSTPM ,
INSTPM_FORCE_ORDERING < < 16 | INSTPM_FORCE_ORDERING ) ;
}
2010-05-21 09:08:55 +08:00
return ret ;
}
2010-12-15 09:56:50 +00:00
static void render_ring_cleanup ( struct intel_ring_buffer * ring )
{
if ( ! ring - > private )
return ;
cleanup_pipe_control ( ring ) ;
}
2010-12-04 11:30:53 +00:00
static void
2011-09-14 20:32:47 -07:00
update_mboxes ( struct intel_ring_buffer * ring ,
u32 seqno ,
u32 mmio_offset )
2010-12-04 11:30:53 +00:00
{
2011-09-14 20:32:47 -07:00
intel_ring_emit ( ring , MI_SEMAPHORE_MBOX |
MI_SEMAPHORE_GLOBAL_GTT |
MI_SEMAPHORE_REGISTER |
MI_SEMAPHORE_UPDATE ) ;
2010-12-04 11:30:53 +00:00
intel_ring_emit ( ring , seqno ) ;
2011-09-14 20:32:47 -07:00
intel_ring_emit ( ring , mmio_offset ) ;
2010-12-04 11:30:53 +00:00
}
2011-09-14 20:32:47 -07:00
/**
* gen6_add_request - Update the semaphore mailbox registers
*
* @ ring - ring that is adding a request
* @ seqno - return seqno stuck into the ring
*
* Update the mailbox registers in the * other * rings with the current seqno .
* This acts like a signal in the canonical semaphore .
*/
2010-12-04 11:30:53 +00:00
static int
gen6_add_request ( struct intel_ring_buffer * ring ,
2011-09-14 20:32:47 -07:00
u32 * seqno )
2010-12-04 11:30:53 +00:00
{
2011-09-14 20:32:47 -07:00
u32 mbox1_reg ;
u32 mbox2_reg ;
2010-12-04 11:30:53 +00:00
int ret ;
ret = intel_ring_begin ( ring , 10 ) ;
if ( ret )
return ret ;
2011-09-14 20:32:47 -07:00
mbox1_reg = ring - > signal_mbox [ 0 ] ;
mbox2_reg = ring - > signal_mbox [ 1 ] ;
2010-12-04 11:30:53 +00:00
2012-01-25 16:32:49 +01:00
* seqno = i915_gem_next_request_seqno ( ring ) ;
2011-09-14 20:32:47 -07:00
update_mboxes ( ring , * seqno , mbox1_reg ) ;
update_mboxes ( ring , * seqno , mbox2_reg ) ;
2010-12-04 11:30:53 +00:00
intel_ring_emit ( ring , MI_STORE_DWORD_INDEX ) ;
intel_ring_emit ( ring , I915_GEM_HWS_INDEX < < MI_STORE_DWORD_INDEX_SHIFT ) ;
2011-09-14 20:32:47 -07:00
intel_ring_emit ( ring , * seqno ) ;
2010-12-04 11:30:53 +00:00
intel_ring_emit ( ring , MI_USER_INTERRUPT ) ;
intel_ring_advance ( ring ) ;
return 0 ;
}
2011-09-14 20:32:47 -07:00
/**
* intel_ring_sync - sync the waiter to the signaller on seqno
*
* @ waiter - ring that is waiting
* @ signaller - ring which has , or will signal
* @ seqno - seqno which the waiter will block on
*/
static int
intel_ring_sync ( struct intel_ring_buffer * waiter ,
struct intel_ring_buffer * signaller ,
int ring ,
2010-12-04 11:30:53 +00:00
u32 seqno )
{
int ret ;
2011-09-14 20:32:47 -07:00
u32 dw1 = MI_SEMAPHORE_MBOX |
MI_SEMAPHORE_COMPARE |
MI_SEMAPHORE_REGISTER ;
2010-12-04 11:30:53 +00:00
2011-09-14 20:32:47 -07:00
ret = intel_ring_begin ( waiter , 4 ) ;
2010-12-04 11:30:53 +00:00
if ( ret )
return ret ;
2011-09-14 20:32:47 -07:00
intel_ring_emit ( waiter , dw1 | signaller - > semaphore_register [ ring ] ) ;
intel_ring_emit ( waiter , seqno ) ;
intel_ring_emit ( waiter , 0 ) ;
intel_ring_emit ( waiter , MI_NOOP ) ;
intel_ring_advance ( waiter ) ;
2010-12-04 11:30:53 +00:00
return 0 ;
}
2011-09-14 20:32:47 -07:00
/* VCS->RCS (RVSYNC) or BCS->RCS (RBSYNC) */
int
render_ring_sync_to ( struct intel_ring_buffer * waiter ,
struct intel_ring_buffer * signaller ,
u32 seqno )
{
WARN_ON ( signaller - > semaphore_register [ RCS ] = = MI_SEMAPHORE_SYNC_INVALID ) ;
return intel_ring_sync ( waiter ,
signaller ,
RCS ,
seqno ) ;
}
/* RCS->VCS (VRSYNC) or BCS->VCS (VBSYNC) */
int
gen6_bsd_ring_sync_to ( struct intel_ring_buffer * waiter ,
struct intel_ring_buffer * signaller ,
u32 seqno )
{
WARN_ON ( signaller - > semaphore_register [ VCS ] = = MI_SEMAPHORE_SYNC_INVALID ) ;
return intel_ring_sync ( waiter ,
signaller ,
VCS ,
seqno ) ;
}
/* RCS->BCS (BRSYNC) or VCS->BCS (BVSYNC) */
int
gen6_blt_ring_sync_to ( struct intel_ring_buffer * waiter ,
struct intel_ring_buffer * signaller ,
u32 seqno )
{
WARN_ON ( signaller - > semaphore_register [ BCS ] = = MI_SEMAPHORE_SYNC_INVALID ) ;
return intel_ring_sync ( waiter ,
signaller ,
BCS ,
seqno ) ;
}
2010-12-15 09:56:50 +00:00
# define PIPE_CONTROL_FLUSH(ring__, addr__) \
do { \
2011-10-11 23:41:08 +02:00
intel_ring_emit ( ring__ , GFX_OP_PIPE_CONTROL ( 4 ) | PIPE_CONTROL_QW_WRITE | \
PIPE_CONTROL_DEPTH_STALL ) ; \
2010-12-15 09:56:50 +00:00
intel_ring_emit ( ring__ , ( addr__ ) | PIPE_CONTROL_GLOBAL_GTT ) ; \
intel_ring_emit ( ring__ , 0 ) ; \
intel_ring_emit ( ring__ , 0 ) ; \
} while ( 0 )
static int
pc_render_add_request ( struct intel_ring_buffer * ring ,
u32 * result )
{
2012-01-25 16:32:49 +01:00
u32 seqno = i915_gem_next_request_seqno ( ring ) ;
2010-12-15 09:56:50 +00:00
struct pipe_control * pc = ring - > private ;
u32 scratch_addr = pc - > gtt_offset + 128 ;
int ret ;
/* For Ironlake, MI_USER_INTERRUPT was deprecated and apparently
* incoherent with writes to memory , i . e . completely fubar ,
* so we need to use PIPE_NOTIFY instead .
*
* However , we also need to workaround the qword write
* incoherence by flushing the 6 PIPE_NOTIFY buffers out to
* memory before requesting an interrupt .
*/
ret = intel_ring_begin ( ring , 32 ) ;
if ( ret )
return ret ;
2011-10-11 23:41:08 +02:00
intel_ring_emit ( ring , GFX_OP_PIPE_CONTROL ( 4 ) | PIPE_CONTROL_QW_WRITE |
2011-10-11 23:41:09 +02:00
PIPE_CONTROL_WRITE_FLUSH |
PIPE_CONTROL_TEXTURE_CACHE_INVALIDATE ) ;
2010-12-15 09:56:50 +00:00
intel_ring_emit ( ring , pc - > gtt_offset | PIPE_CONTROL_GLOBAL_GTT ) ;
intel_ring_emit ( ring , seqno ) ;
intel_ring_emit ( ring , 0 ) ;
PIPE_CONTROL_FLUSH ( ring , scratch_addr ) ;
scratch_addr + = 128 ; /* write to separate cachelines */
PIPE_CONTROL_FLUSH ( ring , scratch_addr ) ;
scratch_addr + = 128 ;
PIPE_CONTROL_FLUSH ( ring , scratch_addr ) ;
scratch_addr + = 128 ;
PIPE_CONTROL_FLUSH ( ring , scratch_addr ) ;
scratch_addr + = 128 ;
PIPE_CONTROL_FLUSH ( ring , scratch_addr ) ;
scratch_addr + = 128 ;
PIPE_CONTROL_FLUSH ( ring , scratch_addr ) ;
2011-10-11 23:41:08 +02:00
intel_ring_emit ( ring , GFX_OP_PIPE_CONTROL ( 4 ) | PIPE_CONTROL_QW_WRITE |
2011-10-11 23:41:09 +02:00
PIPE_CONTROL_WRITE_FLUSH |
PIPE_CONTROL_TEXTURE_CACHE_INVALIDATE |
2010-12-15 09:56:50 +00:00
PIPE_CONTROL_NOTIFY ) ;
intel_ring_emit ( ring , pc - > gtt_offset | PIPE_CONTROL_GLOBAL_GTT ) ;
intel_ring_emit ( ring , seqno ) ;
intel_ring_emit ( ring , 0 ) ;
intel_ring_advance ( ring ) ;
* result = seqno ;
return 0 ;
}
2010-12-04 11:30:53 +00:00
static int
render_ring_add_request ( struct intel_ring_buffer * ring ,
u32 * result )
{
2012-01-25 16:32:49 +01:00
u32 seqno = i915_gem_next_request_seqno ( ring ) ;
2010-12-04 11:30:53 +00:00
int ret ;
2010-10-27 16:11:02 +01:00
2010-12-04 11:30:53 +00:00
ret = intel_ring_begin ( ring , 4 ) ;
if ( ret )
return ret ;
2010-10-27 16:11:02 +01:00
2010-12-04 11:30:53 +00:00
intel_ring_emit ( ring , MI_STORE_DWORD_INDEX ) ;
intel_ring_emit ( ring , I915_GEM_HWS_INDEX < < MI_STORE_DWORD_INDEX_SHIFT ) ;
intel_ring_emit ( ring , seqno ) ;
intel_ring_emit ( ring , MI_USER_INTERRUPT ) ;
2010-10-27 16:11:02 +01:00
intel_ring_advance ( ring ) ;
2010-12-04 11:30:53 +00:00
2010-10-27 16:11:02 +01:00
* result = seqno ;
return 0 ;
2010-05-21 13:26:39 -07:00
}
2012-12-14 16:01:25 +01:00
static u32
gen6_ring_get_seqno ( struct intel_ring_buffer * ring )
{
struct drm_device * dev = ring - > dev ;
/* Workaround to force correct ordering between irq and seqno writes on
* ivb ( and maybe also on snb ) by reading from a CS register ( like
* ACTHD ) before reading the status page . */
if ( IS_GEN7 ( dev ) )
intel_ring_get_active_head ( ring ) ;
return intel_read_status_page ( ring , I915_GEM_HWS_INDEX ) ;
}
2010-05-21 09:08:55 +08:00
static u32
2010-12-04 11:30:53 +00:00
ring_get_seqno ( struct intel_ring_buffer * ring )
2010-05-21 09:08:55 +08:00
{
2010-12-04 11:30:53 +00:00
return intel_read_status_page ( ring , I915_GEM_HWS_INDEX ) ;
}
2010-12-15 09:56:50 +00:00
static u32
pc_render_get_seqno ( struct intel_ring_buffer * ring )
{
struct pipe_control * pc = ring - > private ;
return pc - > cpu_page [ 0 ] ;
}
2011-01-04 17:35:21 +00:00
static void
ironlake_enable_irq ( drm_i915_private_t * dev_priv , u32 mask )
{
dev_priv - > gt_irq_mask & = ~ mask ;
I915_WRITE ( GTIMR , dev_priv - > gt_irq_mask ) ;
POSTING_READ ( GTIMR ) ;
}
static void
ironlake_disable_irq ( drm_i915_private_t * dev_priv , u32 mask )
{
dev_priv - > gt_irq_mask | = mask ;
I915_WRITE ( GTIMR , dev_priv - > gt_irq_mask ) ;
POSTING_READ ( GTIMR ) ;
}
static void
i915_enable_irq ( drm_i915_private_t * dev_priv , u32 mask )
{
dev_priv - > irq_mask & = ~ mask ;
I915_WRITE ( IMR , dev_priv - > irq_mask ) ;
POSTING_READ ( IMR ) ;
}
static void
i915_disable_irq ( drm_i915_private_t * dev_priv , u32 mask )
{
dev_priv - > irq_mask | = mask ;
I915_WRITE ( IMR , dev_priv - > irq_mask ) ;
POSTING_READ ( IMR ) ;
}
2010-12-13 16:54:50 +00:00
static bool
2010-12-04 11:30:53 +00:00
render_ring_get_irq ( struct intel_ring_buffer * ring )
2010-05-21 13:26:39 -07:00
{
2010-10-27 12:18:21 +01:00
struct drm_device * dev = ring - > dev ;
2011-01-04 22:22:56 +00:00
drm_i915_private_t * dev_priv = dev - > dev_private ;
2010-05-21 13:26:39 -07:00
2010-12-13 16:54:50 +00:00
if ( ! dev - > irq_enabled )
return false ;
2011-01-05 10:32:24 +00:00
spin_lock ( & ring - > irq_lock ) ;
2011-01-04 22:22:56 +00:00
if ( ring - > irq_refcount + + = = 0 ) {
2010-05-21 13:26:39 -07:00
if ( HAS_PCH_SPLIT ( dev ) )
2011-01-04 17:35:21 +00:00
ironlake_enable_irq ( dev_priv ,
GT_PIPE_NOTIFY | GT_USER_INTERRUPT ) ;
2010-05-21 13:26:39 -07:00
else
i915_enable_irq ( dev_priv , I915_USER_INTERRUPT ) ;
}
2011-01-05 10:32:24 +00:00
spin_unlock ( & ring - > irq_lock ) ;
2010-12-13 16:54:50 +00:00
return true ;
2010-05-21 13:26:39 -07:00
}
2010-05-21 09:08:55 +08:00
static void
2010-12-04 11:30:53 +00:00
render_ring_put_irq ( struct intel_ring_buffer * ring )
2010-05-21 13:26:39 -07:00
{
2010-10-27 12:18:21 +01:00
struct drm_device * dev = ring - > dev ;
2011-01-04 22:22:56 +00:00
drm_i915_private_t * dev_priv = dev - > dev_private ;
2010-05-21 13:26:39 -07:00
2011-01-05 10:32:24 +00:00
spin_lock ( & ring - > irq_lock ) ;
2011-01-04 22:22:56 +00:00
if ( - - ring - > irq_refcount = = 0 ) {
2010-05-21 13:26:39 -07:00
if ( HAS_PCH_SPLIT ( dev ) )
2011-01-04 17:35:21 +00:00
ironlake_disable_irq ( dev_priv ,
GT_USER_INTERRUPT |
GT_PIPE_NOTIFY ) ;
2010-05-21 13:26:39 -07:00
else
i915_disable_irq ( dev_priv , I915_USER_INTERRUPT ) ;
}
2011-01-05 10:32:24 +00:00
spin_unlock ( & ring - > irq_lock ) ;
2010-05-21 13:26:39 -07:00
}
2010-10-27 12:18:21 +01:00
void intel_ring_setup_status_page ( struct intel_ring_buffer * ring )
2010-05-21 09:08:55 +08:00
{
2011-05-06 17:12:35 -07:00
struct drm_device * dev = ring - > dev ;
2010-10-27 12:18:21 +01:00
drm_i915_private_t * dev_priv = ring - > dev - > dev_private ;
2011-05-06 17:12:35 -07:00
u32 mmio = 0 ;
/* The ring status page addresses are no longer next to the rest of
* the ring registers as of gen7 .
*/
if ( IS_GEN7 ( dev ) ) {
switch ( ring - > id ) {
2011-12-14 13:57:00 +01:00
case RCS :
2011-05-06 17:12:35 -07:00
mmio = RENDER_HWS_PGA_GEN7 ;
break ;
2011-12-14 13:57:00 +01:00
case BCS :
2011-05-06 17:12:35 -07:00
mmio = BLT_HWS_PGA_GEN7 ;
break ;
2011-12-14 13:57:00 +01:00
case VCS :
2011-05-06 17:12:35 -07:00
mmio = BSD_HWS_PGA_GEN7 ;
break ;
}
} else if ( IS_GEN6 ( ring - > dev ) ) {
mmio = RING_HWS_PGA_GEN6 ( ring - > mmio_base ) ;
} else {
mmio = RING_HWS_PGA ( ring - > mmio_base ) ;
}
2010-10-27 12:18:21 +01:00
I915_WRITE ( mmio , ( u32 ) ring - > status_page . gfx_addr ) ;
POSTING_READ ( mmio ) ;
2010-05-21 09:08:55 +08:00
}
2011-01-04 17:34:02 +00:00
static int
2010-10-27 12:18:21 +01:00
bsd_ring_flush ( struct intel_ring_buffer * ring ,
u32 invalidate_domains ,
u32 flush_domains )
2010-05-21 09:08:57 +08:00
{
2011-01-04 17:34:02 +00:00
int ret ;
ret = intel_ring_begin ( ring , 2 ) ;
if ( ret )
return ret ;
intel_ring_emit ( ring , MI_FLUSH ) ;
intel_ring_emit ( ring , MI_NOOP ) ;
intel_ring_advance ( ring ) ;
return 0 ;
2010-05-21 09:08:57 +08:00
}
2010-10-27 16:11:02 +01:00
static int
2010-10-27 12:18:21 +01:00
ring_add_request ( struct intel_ring_buffer * ring ,
2010-10-27 16:11:02 +01:00
u32 * result )
2010-05-21 09:08:57 +08:00
{
u32 seqno ;
2010-10-27 16:11:02 +01:00
int ret ;
ret = intel_ring_begin ( ring , 4 ) ;
if ( ret )
return ret ;
2010-08-07 11:01:22 +01:00
2012-01-25 16:32:49 +01:00
seqno = i915_gem_next_request_seqno ( ring ) ;
2010-08-07 11:01:22 +01:00
2010-10-27 16:11:02 +01:00
intel_ring_emit ( ring , MI_STORE_DWORD_INDEX ) ;
intel_ring_emit ( ring , I915_GEM_HWS_INDEX < < MI_STORE_DWORD_INDEX_SHIFT ) ;
intel_ring_emit ( ring , seqno ) ;
intel_ring_emit ( ring , MI_USER_INTERRUPT ) ;
intel_ring_advance ( ring ) ;
2010-05-21 09:08:57 +08:00
2010-10-27 16:11:02 +01:00
* result = seqno ;
return 0 ;
2010-05-21 09:08:57 +08:00
}
2011-01-04 17:35:21 +00:00
static bool
gen6_ring_get_irq ( struct intel_ring_buffer * ring , u32 gflag , u32 rflag )
{
struct drm_device * dev = ring - > dev ;
2011-01-04 22:22:56 +00:00
drm_i915_private_t * dev_priv = dev - > dev_private ;
2011-01-04 17:35:21 +00:00
if ( ! dev - > irq_enabled )
return false ;
2012-12-14 16:01:25 +01:00
/* It looks like we need to prevent the gt from suspending while waiting
* for an notifiy irq , otherwise irqs seem to get lost on at least the
* blt / bsd rings on ivb . */
2012-01-25 14:04:00 +01:00
gen6_gt_force_wake_get ( dev_priv ) ;
2012-12-14 16:01:25 +01:00
2011-01-05 10:32:24 +00:00
spin_lock ( & ring - > irq_lock ) ;
2011-01-04 22:22:56 +00:00
if ( ring - > irq_refcount + + = = 0 ) {
2011-01-04 17:35:21 +00:00
ring - > irq_mask & = ~ rflag ;
I915_WRITE_IMR ( ring , ring - > irq_mask ) ;
ironlake_enable_irq ( dev_priv , gflag ) ;
}
2011-01-05 10:32:24 +00:00
spin_unlock ( & ring - > irq_lock ) ;
2011-01-04 17:35:21 +00:00
return true ;
}
static void
gen6_ring_put_irq ( struct intel_ring_buffer * ring , u32 gflag , u32 rflag )
{
struct drm_device * dev = ring - > dev ;
2011-01-04 22:22:56 +00:00
drm_i915_private_t * dev_priv = dev - > dev_private ;
2011-01-04 17:35:21 +00:00
2011-01-05 10:32:24 +00:00
spin_lock ( & ring - > irq_lock ) ;
2011-01-04 22:22:56 +00:00
if ( - - ring - > irq_refcount = = 0 ) {
2011-01-04 17:35:21 +00:00
ring - > irq_mask | = rflag ;
I915_WRITE_IMR ( ring , ring - > irq_mask ) ;
ironlake_disable_irq ( dev_priv , gflag ) ;
2010-12-04 11:30:53 +00:00
}
2011-01-05 10:32:24 +00:00
spin_unlock ( & ring - > irq_lock ) ;
2012-12-14 16:01:25 +01:00
2012-01-25 14:04:00 +01:00
gen6_gt_force_wake_put ( dev_priv ) ;
2010-05-21 09:08:57 +08:00
}
2010-12-13 16:54:50 +00:00
static bool
2010-12-04 11:30:53 +00:00
bsd_ring_get_irq ( struct intel_ring_buffer * ring )
2010-05-21 09:08:57 +08:00
{
2011-05-16 16:02:39 +08:00
struct drm_device * dev = ring - > dev ;
drm_i915_private_t * dev_priv = dev - > dev_private ;
if ( ! dev - > irq_enabled )
return false ;
spin_lock ( & ring - > irq_lock ) ;
if ( ring - > irq_refcount + + = = 0 ) {
if ( IS_G4X ( dev ) )
i915_enable_irq ( dev_priv , I915_BSD_USER_INTERRUPT ) ;
else
ironlake_enable_irq ( dev_priv , GT_BSD_USER_INTERRUPT ) ;
}
spin_unlock ( & ring - > irq_lock ) ;
return true ;
2010-12-04 11:30:53 +00:00
}
static void
bsd_ring_put_irq ( struct intel_ring_buffer * ring )
{
2011-05-16 16:02:39 +08:00
struct drm_device * dev = ring - > dev ;
drm_i915_private_t * dev_priv = dev - > dev_private ;
spin_lock ( & ring - > irq_lock ) ;
if ( - - ring - > irq_refcount = = 0 ) {
if ( IS_G4X ( dev ) )
i915_disable_irq ( dev_priv , I915_BSD_USER_INTERRUPT ) ;
else
ironlake_disable_irq ( dev_priv , GT_BSD_USER_INTERRUPT ) ;
}
spin_unlock ( & ring - > irq_lock ) ;
2010-05-21 09:08:57 +08:00
}
static int
2010-11-30 14:10:25 +00:00
ring_dispatch_execbuffer ( struct intel_ring_buffer * ring , u32 offset , u32 length )
2010-05-21 09:08:57 +08:00
{
2010-10-27 12:45:26 +01:00
int ret ;
2010-10-27 12:18:21 +01:00
2010-10-27 12:45:26 +01:00
ret = intel_ring_begin ( ring , 2 ) ;
if ( ret )
return ret ;
2010-10-27 12:18:21 +01:00
intel_ring_emit ( ring ,
2010-11-30 14:10:25 +00:00
MI_BATCH_BUFFER_START | ( 2 < < 6 ) |
2010-10-27 12:18:21 +01:00
MI_BATCH_NON_SECURE_I965 ) ;
2010-11-30 14:10:25 +00:00
intel_ring_emit ( ring , offset ) ;
2010-10-27 12:18:21 +01:00
intel_ring_advance ( ring ) ;
2010-05-21 09:08:57 +08:00
return 0 ;
}
2010-05-21 09:08:55 +08:00
static int
2010-10-27 12:18:21 +01:00
render_ring_dispatch_execbuffer ( struct intel_ring_buffer * ring ,
2010-11-30 14:10:25 +00:00
u32 offset , u32 len )
2010-05-21 13:26:39 -07:00
{
2010-10-27 12:18:21 +01:00
struct drm_device * dev = ring - > dev ;
2010-11-30 14:10:25 +00:00
int ret ;
2010-05-21 13:26:39 -07:00
2010-11-30 14:10:25 +00:00
if ( IS_I830 ( dev ) | | IS_845G ( dev ) ) {
ret = intel_ring_begin ( ring , 4 ) ;
if ( ret )
return ret ;
2010-05-21 13:26:39 -07:00
2010-11-30 14:10:25 +00:00
intel_ring_emit ( ring , MI_BATCH_BUFFER ) ;
intel_ring_emit ( ring , offset | MI_BATCH_NON_SECURE ) ;
intel_ring_emit ( ring , offset + len - 8 ) ;
intel_ring_emit ( ring , 0 ) ;
} else {
ret = intel_ring_begin ( ring , 2 ) ;
if ( ret )
return ret ;
2010-10-27 12:45:26 +01:00
2010-11-30 14:10:25 +00:00
if ( INTEL_INFO ( dev ) - > gen > = 4 ) {
intel_ring_emit ( ring ,
MI_BATCH_BUFFER_START | ( 2 < < 6 ) |
MI_BATCH_NON_SECURE_I965 ) ;
intel_ring_emit ( ring , offset ) ;
2010-05-21 13:26:39 -07:00
} else {
2010-11-30 14:10:25 +00:00
intel_ring_emit ( ring ,
MI_BATCH_BUFFER_START | ( 2 < < 6 ) ) ;
intel_ring_emit ( ring , offset | MI_BATCH_NON_SECURE ) ;
2010-05-21 13:26:39 -07:00
}
}
2010-11-30 14:10:25 +00:00
intel_ring_advance ( ring ) ;
2010-05-21 13:26:39 -07:00
return 0 ;
}
2010-10-27 12:18:21 +01:00
static void cleanup_status_page ( struct intel_ring_buffer * ring )
2010-05-21 13:26:39 -07:00
{
2010-10-27 12:18:21 +01:00
drm_i915_private_t * dev_priv = ring - > dev - > dev_private ;
2010-11-08 19:18:58 +00:00
struct drm_i915_gem_object * obj ;
2010-05-21 13:26:39 -07:00
2010-05-21 09:08:55 +08:00
obj = ring - > status_page . obj ;
if ( obj = = NULL )
2010-05-21 13:26:39 -07:00
return ;
2010-11-08 19:18:58 +00:00
kunmap ( obj - > pages [ 0 ] ) ;
2010-05-21 13:26:39 -07:00
i915_gem_object_unpin ( obj ) ;
2010-11-08 19:18:58 +00:00
drm_gem_object_unreference ( & obj - > base ) ;
2010-05-21 09:08:55 +08:00
ring - > status_page . obj = NULL ;
2010-05-21 13:26:39 -07:00
memset ( & dev_priv - > hws_map , 0 , sizeof ( dev_priv - > hws_map ) ) ;
}
2010-10-27 12:18:21 +01:00
static int init_status_page ( struct intel_ring_buffer * ring )
2010-05-21 13:26:39 -07:00
{
2010-10-27 12:18:21 +01:00
struct drm_device * dev = ring - > dev ;
2010-05-21 13:26:39 -07:00
drm_i915_private_t * dev_priv = dev - > dev_private ;
2010-11-08 19:18:58 +00:00
struct drm_i915_gem_object * obj ;
2010-05-21 13:26:39 -07:00
int ret ;
obj = i915_gem_alloc_object ( dev , 4096 ) ;
if ( obj = = NULL ) {
DRM_ERROR ( " Failed to allocate status page \n " ) ;
ret = - ENOMEM ;
goto err ;
}
2011-04-04 09:44:39 +01:00
i915_gem_object_set_cache_level ( obj , I915_CACHE_LLC ) ;
2010-05-21 13:26:39 -07:00
2010-11-04 17:11:09 +01:00
ret = i915_gem_object_pin ( obj , 4096 , true ) ;
2010-05-21 13:26:39 -07:00
if ( ret ! = 0 ) {
goto err_unref ;
}
2010-11-08 19:18:58 +00:00
ring - > status_page . gfx_addr = obj - > gtt_offset ;
ring - > status_page . page_addr = kmap ( obj - > pages [ 0 ] ) ;
2010-05-21 09:08:55 +08:00
if ( ring - > status_page . page_addr = = NULL ) {
2010-05-21 13:26:39 -07:00
memset ( & dev_priv - > hws_map , 0 , sizeof ( dev_priv - > hws_map ) ) ;
goto err_unpin ;
}
2010-05-21 09:08:55 +08:00
ring - > status_page . obj = obj ;
memset ( ring - > status_page . page_addr , 0 , PAGE_SIZE ) ;
2010-05-21 13:26:39 -07:00
2010-10-27 12:18:21 +01:00
intel_ring_setup_status_page ( ring ) ;
2010-05-21 09:08:55 +08:00
DRM_DEBUG_DRIVER ( " %s hws offset: 0x%08x \n " ,
ring - > name , ring - > status_page . gfx_addr ) ;
2010-05-21 13:26:39 -07:00
return 0 ;
err_unpin :
i915_gem_object_unpin ( obj ) ;
err_unref :
2010-11-08 19:18:58 +00:00
drm_gem_object_unreference ( & obj - > base ) ;
2010-05-21 13:26:39 -07:00
err :
2010-05-21 09:08:55 +08:00
return ret ;
2010-05-21 13:26:39 -07:00
}
2010-05-21 09:08:55 +08:00
int intel_init_ring_buffer ( struct drm_device * dev ,
2010-09-19 17:53:44 +01:00
struct intel_ring_buffer * ring )
2010-05-21 13:26:39 -07:00
{
2010-11-08 19:18:58 +00:00
struct drm_i915_gem_object * obj ;
2010-08-07 11:01:34 +01:00
int ret ;
2010-05-21 09:08:55 +08:00
ring - > dev = dev ;
2010-09-29 16:10:57 +01:00
INIT_LIST_HEAD ( & ring - > active_list ) ;
INIT_LIST_HEAD ( & ring - > request_list ) ;
2010-10-24 12:38:05 +01:00
INIT_LIST_HEAD ( & ring - > gpu_write_list ) ;
2011-01-05 10:32:24 +00:00
2011-03-29 13:19:09 +01:00
init_waitqueue_head ( & ring - > irq_queue ) ;
2011-01-05 10:32:24 +00:00
spin_lock_init ( & ring - > irq_lock ) ;
2011-01-04 17:35:21 +00:00
ring - > irq_mask = ~ 0 ;
2010-05-21 13:26:39 -07:00
2010-05-21 09:08:55 +08:00
if ( I915_NEED_GFX_HWS ( dev ) ) {
2010-10-27 12:18:21 +01:00
ret = init_status_page ( ring ) ;
2010-05-21 09:08:55 +08:00
if ( ret )
return ret ;
}
2010-05-21 13:26:39 -07:00
2010-05-21 09:08:55 +08:00
obj = i915_gem_alloc_object ( dev , ring - > size ) ;
2010-05-21 13:26:39 -07:00
if ( obj = = NULL ) {
DRM_ERROR ( " Failed to allocate ringbuffer \n " ) ;
2010-05-21 09:08:55 +08:00
ret = - ENOMEM ;
2010-08-07 11:01:34 +01:00
goto err_hws ;
2010-05-21 13:26:39 -07:00
}
2010-11-08 19:18:58 +00:00
ring - > obj = obj ;
2010-05-21 09:08:55 +08:00
2010-11-04 17:11:09 +01:00
ret = i915_gem_object_pin ( obj , PAGE_SIZE , true ) ;
2010-08-07 11:01:34 +01:00
if ( ret )
goto err_unref ;
2010-05-21 13:26:39 -07:00
2010-05-21 09:08:55 +08:00
ring - > map . size = ring - > size ;
2010-11-08 19:18:58 +00:00
ring - > map . offset = dev - > agp - > base + obj - > gtt_offset ;
2010-05-21 13:26:39 -07:00
ring - > map . type = 0 ;
ring - > map . flags = 0 ;
ring - > map . mtrr = 0 ;
drm_core_ioremap_wc ( & ring - > map , dev ) ;
if ( ring - > map . handle = = NULL ) {
DRM_ERROR ( " Failed to map ringbuffer. \n " ) ;
2010-05-21 09:08:55 +08:00
ret = - EINVAL ;
2010-08-07 11:01:34 +01:00
goto err_unpin ;
2010-05-21 13:26:39 -07:00
}
2010-05-21 09:08:55 +08:00
ring - > virtual_start = ring - > map . handle ;
2010-10-27 12:18:21 +01:00
ret = ring - > init ( ring ) ;
2010-08-07 11:01:34 +01:00
if ( ret )
goto err_unmap ;
2010-05-21 13:26:39 -07:00
2010-12-22 14:04:47 +00:00
/* Workaround an erratum on the i830 which causes a hang if
* the TAIL pointer points to within the last 2 cachelines
* of the buffer .
*/
ring - > effective_size = ring - > size ;
if ( IS_I830 ( ring - > dev ) )
ring - > effective_size - = 128 ;
2010-10-29 18:15:52 +01:00
return 0 ;
2010-08-07 11:01:34 +01:00
err_unmap :
drm_core_ioremapfree ( & ring - > map , dev ) ;
err_unpin :
i915_gem_object_unpin ( obj ) ;
err_unref :
2010-11-08 19:18:58 +00:00
drm_gem_object_unreference ( & obj - > base ) ;
ring - > obj = NULL ;
2010-08-07 11:01:34 +01:00
err_hws :
2010-10-27 12:18:21 +01:00
cleanup_status_page ( ring ) ;
2010-05-21 09:08:55 +08:00
return ret ;
2010-05-21 13:26:39 -07:00
}
2010-10-27 12:18:21 +01:00
void intel_cleanup_ring_buffer ( struct intel_ring_buffer * ring )
2010-05-21 13:26:39 -07:00
{
2010-10-29 16:18:36 +01:00
struct drm_i915_private * dev_priv ;
int ret ;
2010-11-08 19:18:58 +00:00
if ( ring - > obj = = NULL )
2010-05-21 13:26:39 -07:00
return ;
2010-10-29 16:18:36 +01:00
/* Disable the ring buffer. The ring must be idle at this point */
dev_priv = ring - > dev - > dev_private ;
2011-03-19 18:14:27 -07:00
ret = intel_wait_ring_idle ( ring ) ;
2011-01-24 16:35:42 +00:00
if ( ret )
DRM_ERROR ( " failed to quiesce %s whilst cleaning up: %d \n " ,
ring - > name , ret ) ;
2010-10-29 16:18:36 +01:00
I915_WRITE_CTL ( ring , 0 ) ;
2010-10-27 12:18:21 +01:00
drm_core_ioremapfree ( & ring - > map , ring - > dev ) ;
2010-05-21 13:26:39 -07:00
2010-11-08 19:18:58 +00:00
i915_gem_object_unpin ( ring - > obj ) ;
drm_gem_object_unreference ( & ring - > obj - > base ) ;
ring - > obj = NULL ;
2010-10-27 12:18:21 +01:00
2010-11-02 16:31:01 +08:00
if ( ring - > cleanup )
ring - > cleanup ( ring ) ;
2010-10-27 12:18:21 +01:00
cleanup_status_page ( ring ) ;
2010-05-21 13:26:39 -07:00
}
2010-10-27 12:18:21 +01:00
static int intel_wrap_ring_buffer ( struct intel_ring_buffer * ring )
2010-05-21 13:26:39 -07:00
{
2010-05-21 09:08:55 +08:00
unsigned int * virt ;
2010-12-22 14:04:47 +00:00
int rem = ring - > size - ring - > tail ;
2010-05-21 13:26:39 -07:00
2010-05-21 09:08:55 +08:00
if ( ring - > space < rem ) {
2010-10-27 12:18:21 +01:00
int ret = intel_wait_ring_buffer ( ring , rem ) ;
2010-05-21 13:26:39 -07:00
if ( ret )
return ret ;
}
2010-05-21 09:08:55 +08:00
virt = ( unsigned int * ) ( ring - > virtual_start + ring - > tail ) ;
2010-08-04 15:18:12 +01:00
rem / = 8 ;
while ( rem - - ) {
2010-05-21 13:26:39 -07:00
* virt + + = MI_NOOP ;
2010-08-04 15:18:12 +01:00
* virt + + = MI_NOOP ;
}
2010-05-21 13:26:39 -07:00
2010-05-21 09:08:55 +08:00
ring - > tail = 0 ;
2011-01-20 17:00:10 +00:00
ring - > space = ring_space ( ring ) ;
2010-05-21 13:26:39 -07:00
return 0 ;
}
2010-10-27 12:18:21 +01:00
int intel_wait_ring_buffer ( struct intel_ring_buffer * ring , int n )
2010-05-21 13:26:39 -07:00
{
2010-10-27 12:18:21 +01:00
struct drm_device * dev = ring - > dev ;
2010-11-09 17:17:32 +08:00
struct drm_i915_private * dev_priv = dev - > dev_private ;
2010-10-27 12:18:21 +01:00
unsigned long end ;
2010-10-29 21:44:37 +01:00
u32 head ;
2011-01-20 17:00:10 +00:00
/* If the reported head position has wrapped or hasn't advanced,
* fallback to the slow and accurate path .
*/
head = intel_read_status_page ( ring , 4 ) ;
if ( head > ring - > head ) {
ring - > head = head ;
ring - > space = ring_space ( ring ) ;
if ( ring - > space > = n )
return 0 ;
}
2011-02-03 11:57:46 +00:00
trace_i915_ring_wait_begin ( ring ) ;
2011-12-14 13:56:59 +01:00
if ( drm_core_check_feature ( dev , DRIVER_GEM ) )
/* With GEM the hangcheck timer should kick us out of the loop,
* leaving it early runs the risk of corrupting GEM state ( due
* to running on almost untested codepaths ) . But on resume
* timers don ' t work yet , so prevent a complete hang in that
* case by choosing an insanely large timeout . */
end = jiffies + 60 * HZ ;
else
end = jiffies + 3 * HZ ;
2010-05-21 09:08:55 +08:00
do {
2011-01-20 17:00:10 +00:00
ring - > head = I915_READ_HEAD ( ring ) ;
ring - > space = ring_space ( ring ) ;
2010-05-21 13:26:39 -07:00
if ( ring - > space > = n ) {
2011-02-03 11:57:46 +00:00
trace_i915_ring_wait_end ( ring ) ;
2010-05-21 13:26:39 -07:00
return 0 ;
}
if ( dev - > primary - > master ) {
struct drm_i915_master_private * master_priv = dev - > primary - > master - > driver_priv ;
if ( master_priv - > sarea_priv )
master_priv - > sarea_priv - > perf_boxes | = I915_BOX_WAIT ;
}
2010-05-21 09:08:57 +08:00
2010-10-13 10:09:14 +01:00
msleep ( 1 ) ;
2010-10-29 21:06:16 +01:00
if ( atomic_read ( & dev_priv - > mm . wedged ) )
return - EAGAIN ;
2010-05-21 09:08:55 +08:00
} while ( ! time_after ( jiffies , end ) ) ;
2011-02-03 11:57:46 +00:00
trace_i915_ring_wait_end ( ring ) ;
2010-05-21 09:08:55 +08:00
return - EBUSY ;
}
2010-05-21 13:26:39 -07:00
2010-10-27 12:45:26 +01:00
int intel_ring_begin ( struct intel_ring_buffer * ring ,
int num_dwords )
2010-05-21 09:08:55 +08:00
{
2011-01-26 15:55:56 +00:00
struct drm_i915_private * dev_priv = ring - > dev - > dev_private ;
2010-06-12 17:40:24 +08:00
int n = 4 * num_dwords ;
2010-10-27 12:45:26 +01:00
int ret ;
2010-10-27 12:18:21 +01:00
2011-01-26 15:55:56 +00:00
if ( unlikely ( atomic_read ( & dev_priv - > mm . wedged ) ) )
return - EIO ;
2010-12-22 14:04:47 +00:00
if ( unlikely ( ring - > tail + n > ring - > effective_size ) ) {
2010-10-27 12:45:26 +01:00
ret = intel_wrap_ring_buffer ( ring ) ;
if ( unlikely ( ret ) )
return ret ;
}
2010-10-27 12:18:21 +01:00
2010-10-27 12:45:26 +01:00
if ( unlikely ( ring - > space < n ) ) {
ret = intel_wait_ring_buffer ( ring , n ) ;
if ( unlikely ( ret ) )
return ret ;
}
2010-08-04 15:18:13 +01:00
ring - > space - = n ;
2010-10-27 12:45:26 +01:00
return 0 ;
2010-05-21 09:08:55 +08:00
}
2010-05-21 13:26:39 -07:00
2010-10-27 12:18:21 +01:00
void intel_ring_advance ( struct intel_ring_buffer * ring )
2010-05-21 09:08:55 +08:00
{
2010-08-04 15:18:13 +01:00
ring - > tail & = ring - > size - 1 ;
2010-10-27 12:18:21 +01:00
ring - > write_tail ( ring , ring - > tail ) ;
2010-05-21 09:08:55 +08:00
}
2010-05-21 13:26:39 -07:00
2010-09-19 14:46:27 +01:00
static const struct intel_ring_buffer render_ring = {
2010-05-21 09:08:55 +08:00
. name = " render ring " ,
2011-12-14 13:57:00 +01:00
. id = RCS ,
2010-08-02 16:24:01 +02:00
. mmio_base = RENDER_RING_BASE ,
2010-05-21 09:08:55 +08:00
. size = 32 * PAGE_SIZE ,
. init = init_render_ring ,
2010-10-22 17:02:41 +01:00
. write_tail = ring_write_tail ,
2010-05-21 09:08:55 +08:00
. flush = render_ring_flush ,
. add_request = render_ring_add_request ,
2010-12-04 11:30:53 +00:00
. get_seqno = ring_get_seqno ,
. irq_get = render_ring_get_irq ,
. irq_put = render_ring_put_irq ,
2010-10-27 12:18:21 +01:00
. dispatch_execbuffer = render_ring_dispatch_execbuffer ,
2011-08-16 15:34:10 -04:00
. cleanup = render_ring_cleanup ,
2011-09-14 20:32:47 -07:00
. sync_to = render_ring_sync_to ,
. semaphore_register = { MI_SEMAPHORE_SYNC_INVALID ,
MI_SEMAPHORE_SYNC_RV ,
MI_SEMAPHORE_SYNC_RB } ,
. signal_mbox = { GEN6_VRSYNC , GEN6_BRSYNC } ,
2010-05-21 09:08:55 +08:00
} ;
2010-05-21 09:08:57 +08:00
/* ring buffer for bit-stream decoder */
2010-09-19 14:46:27 +01:00
static const struct intel_ring_buffer bsd_ring = {
2010-05-21 09:08:57 +08:00
. name = " bsd ring " ,
2011-12-14 13:57:00 +01:00
. id = VCS ,
2010-08-02 16:24:01 +02:00
. mmio_base = BSD_RING_BASE ,
2010-05-21 09:08:57 +08:00
. size = 32 * PAGE_SIZE ,
2010-10-27 12:18:21 +01:00
. init = init_ring_common ,
2010-10-22 17:02:41 +01:00
. write_tail = ring_write_tail ,
2010-05-21 09:08:57 +08:00
. flush = bsd_ring_flush ,
2010-10-19 11:19:32 +01:00
. add_request = ring_add_request ,
2010-12-04 11:30:53 +00:00
. get_seqno = ring_get_seqno ,
. irq_get = bsd_ring_get_irq ,
. irq_put = bsd_ring_put_irq ,
2010-10-27 12:18:21 +01:00
. dispatch_execbuffer = ring_dispatch_execbuffer ,
2010-05-21 09:08:57 +08:00
} ;
2010-09-16 10:43:11 +08:00
2010-09-19 14:40:43 +01:00
2010-10-27 12:18:21 +01:00
static void gen6_bsd_ring_write_tail ( struct intel_ring_buffer * ring ,
2010-10-22 17:02:41 +01:00
u32 value )
2010-09-19 14:40:43 +01:00
{
2011-08-16 15:34:10 -04:00
drm_i915_private_t * dev_priv = ring - > dev - > dev_private ;
2010-09-19 14:40:43 +01:00
/* Every tail move must follow the sequence below */
2011-08-16 15:34:10 -04:00
I915_WRITE ( GEN6_BSD_SLEEP_PSMI_CONTROL ,
GEN6_BSD_SLEEP_PSMI_CONTROL_RC_ILDL_MESSAGE_MODIFY_MASK |
GEN6_BSD_SLEEP_PSMI_CONTROL_RC_ILDL_MESSAGE_DISABLE ) ;
I915_WRITE ( GEN6_BSD_RNCID , 0x0 ) ;
if ( wait_for ( ( I915_READ ( GEN6_BSD_SLEEP_PSMI_CONTROL ) &
GEN6_BSD_SLEEP_PSMI_CONTROL_IDLE_INDICATOR ) = = 0 ,
50 ) )
DRM_ERROR ( " timed out waiting for IDLE Indicator \n " ) ;
I915_WRITE_TAIL ( ring , value ) ;
I915_WRITE ( GEN6_BSD_SLEEP_PSMI_CONTROL ,
GEN6_BSD_SLEEP_PSMI_CONTROL_RC_ILDL_MESSAGE_MODIFY_MASK |
GEN6_BSD_SLEEP_PSMI_CONTROL_RC_ILDL_MESSAGE_ENABLE ) ;
2010-09-19 14:40:43 +01:00
}
2011-01-04 17:34:02 +00:00
static int gen6_ring_flush ( struct intel_ring_buffer * ring ,
2011-02-02 12:13:49 +00:00
u32 invalidate , u32 flush )
2010-09-19 14:40:43 +01:00
{
2011-02-02 12:13:49 +00:00
uint32_t cmd ;
2011-01-04 17:34:02 +00:00
int ret ;
ret = intel_ring_begin ( ring , 4 ) ;
if ( ret )
return ret ;
2011-02-02 12:13:49 +00:00
cmd = MI_FLUSH_DW ;
if ( invalidate & I915_GEM_GPU_DOMAINS )
cmd | = MI_INVALIDATE_TLB | MI_INVALIDATE_BSD ;
intel_ring_emit ( ring , cmd ) ;
2011-01-04 17:34:02 +00:00
intel_ring_emit ( ring , 0 ) ;
intel_ring_emit ( ring , 0 ) ;
2011-02-02 12:13:49 +00:00
intel_ring_emit ( ring , MI_NOOP ) ;
2011-01-04 17:34:02 +00:00
intel_ring_advance ( ring ) ;
return 0 ;
2010-09-19 14:40:43 +01:00
}
static int
2010-10-27 12:18:21 +01:00
gen6_ring_dispatch_execbuffer ( struct intel_ring_buffer * ring ,
2010-11-30 14:10:25 +00:00
u32 offset , u32 len )
2010-09-19 14:40:43 +01:00
{
2011-08-16 15:34:10 -04:00
int ret ;
2010-09-19 17:53:44 +01:00
2011-08-16 15:34:10 -04:00
ret = intel_ring_begin ( ring , 2 ) ;
if ( ret )
return ret ;
2010-10-27 12:45:26 +01:00
2011-08-16 15:34:10 -04:00
intel_ring_emit ( ring , MI_BATCH_BUFFER_START | MI_BATCH_NON_SECURE_I965 ) ;
/* bit0-7 is the length on GEN6+ */
intel_ring_emit ( ring , offset ) ;
intel_ring_advance ( ring ) ;
2010-09-19 17:53:44 +01:00
2011-08-16 15:34:10 -04:00
return 0 ;
2010-09-19 14:40:43 +01:00
}
2011-01-04 17:35:21 +00:00
static bool
gen6_render_ring_get_irq ( struct intel_ring_buffer * ring )
{
return gen6_ring_get_irq ( ring ,
GT_USER_INTERRUPT ,
GEN6_RENDER_USER_INTERRUPT ) ;
}
static void
gen6_render_ring_put_irq ( struct intel_ring_buffer * ring )
{
return gen6_ring_put_irq ( ring ,
GT_USER_INTERRUPT ,
GEN6_RENDER_USER_INTERRUPT ) ;
}
2010-12-13 16:54:50 +00:00
static bool
2010-12-04 11:30:53 +00:00
gen6_bsd_ring_get_irq ( struct intel_ring_buffer * ring )
{
2011-01-04 17:35:21 +00:00
return gen6_ring_get_irq ( ring ,
GT_GEN6_BSD_USER_INTERRUPT ,
GEN6_BSD_USER_INTERRUPT ) ;
2010-12-04 11:30:53 +00:00
}
static void
gen6_bsd_ring_put_irq ( struct intel_ring_buffer * ring )
{
2011-01-04 17:35:21 +00:00
return gen6_ring_put_irq ( ring ,
GT_GEN6_BSD_USER_INTERRUPT ,
GEN6_BSD_USER_INTERRUPT ) ;
2010-12-04 11:30:53 +00:00
}
2010-09-19 14:40:43 +01:00
/* ring buffer for Video Codec for Gen6+ */
2010-09-19 14:46:27 +01:00
static const struct intel_ring_buffer gen6_bsd_ring = {
2010-12-04 11:30:53 +00:00
. name = " gen6 bsd ring " ,
2011-12-14 13:57:00 +01:00
. id = VCS ,
2010-12-04 11:30:53 +00:00
. mmio_base = GEN6_BSD_RING_BASE ,
. size = 32 * PAGE_SIZE ,
. init = init_ring_common ,
. write_tail = gen6_bsd_ring_write_tail ,
. flush = gen6_ring_flush ,
. add_request = gen6_add_request ,
2012-12-14 16:01:25 +01:00
. get_seqno = gen6_ring_get_seqno ,
2010-12-04 11:30:53 +00:00
. irq_get = gen6_bsd_ring_get_irq ,
. irq_put = gen6_bsd_ring_put_irq ,
. dispatch_execbuffer = gen6_ring_dispatch_execbuffer ,
2011-09-14 20:32:47 -07:00
. sync_to = gen6_bsd_ring_sync_to ,
. semaphore_register = { MI_SEMAPHORE_SYNC_VR ,
MI_SEMAPHORE_SYNC_INVALID ,
MI_SEMAPHORE_SYNC_VB } ,
. signal_mbox = { GEN6_RVSYNC , GEN6_BVSYNC } ,
2010-10-19 11:19:32 +01:00
} ;
/* Blitter support (SandyBridge+) */
2010-12-13 16:54:50 +00:00
static bool
2010-12-04 11:30:53 +00:00
blt_ring_get_irq ( struct intel_ring_buffer * ring )
2010-10-19 11:19:32 +01:00
{
2011-01-04 17:35:21 +00:00
return gen6_ring_get_irq ( ring ,
GT_BLT_USER_INTERRUPT ,
GEN6_BLITTER_USER_INTERRUPT ) ;
2010-10-19 11:19:32 +01:00
}
2010-12-04 11:30:53 +00:00
2010-10-19 11:19:32 +01:00
static void
2010-12-04 11:30:53 +00:00
blt_ring_put_irq ( struct intel_ring_buffer * ring )
2010-10-19 11:19:32 +01:00
{
2011-01-04 17:35:21 +00:00
gen6_ring_put_irq ( ring ,
GT_BLT_USER_INTERRUPT ,
GEN6_BLITTER_USER_INTERRUPT ) ;
2010-10-19 11:19:32 +01:00
}
2011-01-04 17:34:02 +00:00
static int blt_ring_flush ( struct intel_ring_buffer * ring ,
2011-02-02 12:13:49 +00:00
u32 invalidate , u32 flush )
2010-11-02 16:31:01 +08:00
{
2011-02-02 12:13:49 +00:00
uint32_t cmd ;
2011-01-04 17:34:02 +00:00
int ret ;
2011-12-14 13:57:07 +01:00
ret = intel_ring_begin ( ring , 4 ) ;
2011-01-04 17:34:02 +00:00
if ( ret )
return ret ;
2011-02-02 12:13:49 +00:00
cmd = MI_FLUSH_DW ;
if ( invalidate & I915_GEM_DOMAIN_RENDER )
cmd | = MI_INVALIDATE_TLB ;
intel_ring_emit ( ring , cmd ) ;
2011-01-04 17:34:02 +00:00
intel_ring_emit ( ring , 0 ) ;
intel_ring_emit ( ring , 0 ) ;
2011-02-02 12:13:49 +00:00
intel_ring_emit ( ring , MI_NOOP ) ;
2011-01-04 17:34:02 +00:00
intel_ring_advance ( ring ) ;
return 0 ;
2010-11-02 16:31:01 +08:00
}
2010-10-19 11:19:32 +01:00
static const struct intel_ring_buffer gen6_blt_ring = {
2011-08-16 15:34:10 -04:00
. name = " blt ring " ,
2011-12-14 13:57:00 +01:00
. id = BCS ,
2011-08-16 15:34:10 -04:00
. mmio_base = BLT_RING_BASE ,
. size = 32 * PAGE_SIZE ,
2011-12-14 13:57:07 +01:00
. init = init_ring_common ,
2011-08-16 15:34:10 -04:00
. write_tail = ring_write_tail ,
. flush = blt_ring_flush ,
. add_request = gen6_add_request ,
2012-12-14 16:01:25 +01:00
. get_seqno = gen6_ring_get_seqno ,
2011-09-14 20:32:47 -07:00
. irq_get = blt_ring_get_irq ,
. irq_put = blt_ring_put_irq ,
2011-08-16 15:34:10 -04:00
. dispatch_execbuffer = gen6_ring_dispatch_execbuffer ,
2011-09-14 20:32:47 -07:00
. sync_to = gen6_blt_ring_sync_to ,
. semaphore_register = { MI_SEMAPHORE_SYNC_BR ,
MI_SEMAPHORE_SYNC_BV ,
MI_SEMAPHORE_SYNC_INVALID } ,
. signal_mbox = { GEN6_RBSYNC , GEN6_VBSYNC } ,
2010-09-19 14:40:43 +01:00
} ;
2010-09-16 10:43:11 +08:00
int intel_init_render_ring_buffer ( struct drm_device * dev )
{
drm_i915_private_t * dev_priv = dev - > dev_private ;
2010-12-04 11:30:53 +00:00
struct intel_ring_buffer * ring = & dev_priv - > ring [ RCS ] ;
2010-09-16 10:43:11 +08:00
2010-12-04 11:30:53 +00:00
* ring = render_ring ;
if ( INTEL_INFO ( dev ) - > gen > = 6 ) {
ring - > add_request = gen6_add_request ;
2011-10-16 10:23:31 +02:00
ring - > flush = gen6_render_ring_flush ;
2011-01-04 17:35:21 +00:00
ring - > irq_get = gen6_render_ring_get_irq ;
ring - > irq_put = gen6_render_ring_put_irq ;
2012-12-14 16:01:25 +01:00
ring - > get_seqno = gen6_ring_get_seqno ;
2010-12-15 09:56:50 +00:00
} else if ( IS_GEN5 ( dev ) ) {
ring - > add_request = pc_render_add_request ;
ring - > get_seqno = pc_render_get_seqno ;
2010-12-04 11:30:53 +00:00
}
2010-09-16 10:43:11 +08:00
if ( ! I915_NEED_GFX_HWS ( dev ) ) {
2010-12-04 11:30:53 +00:00
ring - > status_page . page_addr = dev_priv - > status_page_dmah - > vaddr ;
memset ( ring - > status_page . page_addr , 0 , PAGE_SIZE ) ;
2010-09-16 10:43:11 +08:00
}
2010-12-04 11:30:53 +00:00
return intel_init_ring_buffer ( dev , ring ) ;
2010-09-16 10:43:11 +08:00
}
2011-01-20 09:57:11 +00:00
int intel_render_ring_init_dri ( struct drm_device * dev , u64 start , u32 size )
{
drm_i915_private_t * dev_priv = dev - > dev_private ;
struct intel_ring_buffer * ring = & dev_priv - > ring [ RCS ] ;
* ring = render_ring ;
if ( INTEL_INFO ( dev ) - > gen > = 6 ) {
ring - > add_request = gen6_add_request ;
ring - > irq_get = gen6_render_ring_get_irq ;
ring - > irq_put = gen6_render_ring_put_irq ;
} else if ( IS_GEN5 ( dev ) ) {
ring - > add_request = pc_render_add_request ;
ring - > get_seqno = pc_render_get_seqno ;
}
2011-07-22 10:44:39 -07:00
if ( ! I915_NEED_GFX_HWS ( dev ) )
ring - > status_page . page_addr = dev_priv - > status_page_dmah - > vaddr ;
2011-01-20 09:57:11 +00:00
ring - > dev = dev ;
INIT_LIST_HEAD ( & ring - > active_list ) ;
INIT_LIST_HEAD ( & ring - > request_list ) ;
INIT_LIST_HEAD ( & ring - > gpu_write_list ) ;
ring - > size = size ;
ring - > effective_size = ring - > size ;
if ( IS_I830 ( ring - > dev ) )
ring - > effective_size - = 128 ;
ring - > map . offset = start ;
ring - > map . size = size ;
ring - > map . type = 0 ;
ring - > map . flags = 0 ;
ring - > map . mtrr = 0 ;
drm_core_ioremap_wc ( & ring - > map , dev ) ;
if ( ring - > map . handle = = NULL ) {
DRM_ERROR ( " can not ioremap virtual address for "
" ring buffer \n " ) ;
return - ENOMEM ;
}
ring - > virtual_start = ( void __force __iomem * ) ring - > map . handle ;
return 0 ;
}
2010-09-16 10:43:11 +08:00
int intel_init_bsd_ring_buffer ( struct drm_device * dev )
{
drm_i915_private_t * dev_priv = dev - > dev_private ;
2010-12-04 11:30:53 +00:00
struct intel_ring_buffer * ring = & dev_priv - > ring [ VCS ] ;
2010-09-16 10:43:11 +08:00
2011-04-06 14:54:44 -07:00
if ( IS_GEN6 ( dev ) | | IS_GEN7 ( dev ) )
2010-12-04 11:30:53 +00:00
* ring = gen6_bsd_ring ;
2010-09-19 14:40:43 +01:00
else
2010-12-04 11:30:53 +00:00
* ring = bsd_ring ;
2010-09-16 10:43:11 +08:00
2010-12-04 11:30:53 +00:00
return intel_init_ring_buffer ( dev , ring ) ;
2010-09-16 10:43:11 +08:00
}
2010-10-19 11:19:32 +01:00
int intel_init_blt_ring_buffer ( struct drm_device * dev )
{
drm_i915_private_t * dev_priv = dev - > dev_private ;
2010-12-04 11:30:53 +00:00
struct intel_ring_buffer * ring = & dev_priv - > ring [ BCS ] ;
2010-10-19 11:19:32 +01:00
2010-12-04 11:30:53 +00:00
* ring = gen6_blt_ring ;
2010-10-19 11:19:32 +01:00
2010-12-04 11:30:53 +00:00
return intel_init_ring_buffer ( dev , ring ) ;
2010-10-19 11:19:32 +01:00
}