via_dma.c

来自「linux 内核源代码」· C语言 代码 · 共 715 行 · 第 1/2 页

C
715
字号
	}	if (DRM_COPY_FROM_USER(dev_priv->pci_buf, cmd->buf, cmd->size))		return -EFAULT;	if ((ret =	     via_verify_command_stream((uint32_t *) dev_priv->pci_buf,				       cmd->size, dev, 0))) {		return ret;	}	ret =	    via_parse_command_stream(dev, (const uint32_t *)dev_priv->pci_buf,				     cmd->size);	return ret;}static int via_pci_cmdbuffer(struct drm_device *dev, void *data, struct drm_file *file_priv){	drm_via_cmdbuffer_t *cmdbuf = data;	int ret;	LOCK_TEST_WITH_RETURN(dev, file_priv);	DRM_DEBUG("via_pci_cmdbuffer, buf %p size %lu\n", cmdbuf->buf,		  cmdbuf->size);	ret = via_dispatch_pci_cmdbuffer(dev, cmdbuf);	if (ret) {		return ret;	}	return 0;}static inline uint32_t *via_align_buffer(drm_via_private_t * dev_priv,					 uint32_t * vb, int qw_count){	for (; qw_count > 0; --qw_count) {		VIA_OUT_RING_QW(HC_DUMMY, HC_DUMMY);	}	return vb;}/* * This function is used internally by ring buffer mangement code. * * Returns virtual pointer to ring buffer. */static inline uint32_t *via_get_dma(drm_via_private_t * dev_priv){	return (uint32_t *) (dev_priv->dma_ptr + dev_priv->dma_low);}/* * Hooks a segment of data into the tail of the ring-buffer by * modifying the pause address stored in the buffer itself. If * the regulator has already paused, restart it. */static int via_hook_segment(drm_via_private_t * dev_priv,			    uint32_t pause_addr_hi, uint32_t pause_addr_lo,			    int no_pci_fire){	int paused, count;	volatile uint32_t *paused_at = dev_priv->last_pause_ptr;	uint32_t reader,ptr;	paused = 0;	via_flush_write_combine();	(void) *(volatile uint32_t *)(via_get_dma(dev_priv) -1);	*paused_at = pause_addr_lo;	via_flush_write_combine();	(void) *paused_at;	reader = *(dev_priv->hw_addr_ptr);	ptr = ((volatile char *)paused_at - dev_priv->dma_ptr) +		dev_priv->dma_offset + (uint32_t) dev_priv->agpAddr + 4;	dev_priv->last_pause_ptr = via_get_dma(dev_priv) - 1;	if ((ptr - reader) <= dev_priv->dma_diff ) {		count = 10000000;		while (!(paused = (VIA_READ(0x41c) & 0x80000000)) && count--);	}	if (paused && !no_pci_fire) {		reader = *(dev_priv->hw_addr_ptr);		if ((ptr - reader) == dev_priv->dma_diff) {			/*			 * There is a concern that these writes may stall the PCI bus			 * if the GPU is not idle. However, idling the GPU first			 * doesn't make a difference.			 */			VIA_WRITE(VIA_REG_TRANSET, (HC_ParaType_PreCR << 16));			VIA_WRITE(VIA_REG_TRANSPACE, pause_addr_hi);			VIA_WRITE(VIA_REG_TRANSPACE, pause_addr_lo);			VIA_READ(VIA_REG_TRANSPACE);		}	}	return paused;}static int via_wait_idle(drm_via_private_t * dev_priv){	int count = 10000000;	while (!(VIA_READ(VIA_REG_STATUS) & VIA_VR_QUEUE_BUSY) && count--);	while (count-- && (VIA_READ(VIA_REG_STATUS) &			   (VIA_CMD_RGTR_BUSY | VIA_2D_ENG_BUSY |			    VIA_3D_ENG_BUSY))) ;	return count;}static uint32_t *via_align_cmd(drm_via_private_t * dev_priv, uint32_t cmd_type,			       uint32_t addr, uint32_t * cmd_addr_hi,			       uint32_t * cmd_addr_lo, int skip_wait){	uint32_t agp_base;	uint32_t cmd_addr, addr_lo, addr_hi;	uint32_t *vb;	uint32_t qw_pad_count;	if (!skip_wait)		via_cmdbuf_wait(dev_priv, 2 * CMDBUF_ALIGNMENT_SIZE);	vb = via_get_dma(dev_priv);	VIA_OUT_RING_QW(HC_HEADER2 | ((VIA_REG_TRANSET >> 2) << 12) |			(VIA_REG_TRANSPACE >> 2), HC_ParaType_PreCR << 16);	agp_base = dev_priv->dma_offset + (uint32_t) dev_priv->agpAddr;	qw_pad_count = (CMDBUF_ALIGNMENT_SIZE >> 3) -	    ((dev_priv->dma_low & CMDBUF_ALIGNMENT_MASK) >> 3);	cmd_addr = (addr) ? addr :	    agp_base + dev_priv->dma_low - 8 + (qw_pad_count << 3);	addr_lo = ((HC_SubA_HAGPBpL << 24) | (cmd_type & HC_HAGPBpID_MASK) |		   (cmd_addr & HC_HAGPBpL_MASK));	addr_hi = ((HC_SubA_HAGPBpH << 24) | (cmd_addr >> 24));	vb = via_align_buffer(dev_priv, vb, qw_pad_count - 1);	VIA_OUT_RING_QW(*cmd_addr_hi = addr_hi, *cmd_addr_lo = addr_lo);	return vb;}static void via_cmdbuf_start(drm_via_private_t * dev_priv){	uint32_t pause_addr_lo, pause_addr_hi;	uint32_t start_addr, start_addr_lo;	uint32_t end_addr, end_addr_lo;	uint32_t command;	uint32_t agp_base;	uint32_t ptr;	uint32_t reader;	int count;	dev_priv->dma_low = 0;	agp_base = dev_priv->dma_offset + (uint32_t) dev_priv->agpAddr;	start_addr = agp_base;	end_addr = agp_base + dev_priv->dma_high;	start_addr_lo = ((HC_SubA_HAGPBstL << 24) | (start_addr & 0xFFFFFF));	end_addr_lo = ((HC_SubA_HAGPBendL << 24) | (end_addr & 0xFFFFFF));	command = ((HC_SubA_HAGPCMNT << 24) | (start_addr >> 24) |		   ((end_addr & 0xff000000) >> 16));	dev_priv->last_pause_ptr =	    via_align_cmd(dev_priv, HC_HAGPBpID_PAUSE, 0,			  &pause_addr_hi, &pause_addr_lo, 1) - 1;	via_flush_write_combine();	(void) *(volatile uint32_t *)dev_priv->last_pause_ptr;	VIA_WRITE(VIA_REG_TRANSET, (HC_ParaType_PreCR << 16));	VIA_WRITE(VIA_REG_TRANSPACE, command);	VIA_WRITE(VIA_REG_TRANSPACE, start_addr_lo);	VIA_WRITE(VIA_REG_TRANSPACE, end_addr_lo);	VIA_WRITE(VIA_REG_TRANSPACE, pause_addr_hi);	VIA_WRITE(VIA_REG_TRANSPACE, pause_addr_lo);	DRM_WRITEMEMORYBARRIER();	VIA_WRITE(VIA_REG_TRANSPACE, command | HC_HAGPCMNT_MASK);	VIA_READ(VIA_REG_TRANSPACE);	dev_priv->dma_diff = 0;	count = 10000000;	while (!(VIA_READ(0x41c) & 0x80000000) && count--);	reader = *(dev_priv->hw_addr_ptr);	ptr = ((volatile char *)dev_priv->last_pause_ptr - dev_priv->dma_ptr) +	    dev_priv->dma_offset + (uint32_t) dev_priv->agpAddr + 4;	/*	 * This is the difference between where we tell the	 * command reader to pause and where it actually pauses.	 * This differs between hw implementation so we need to	 * detect it.	 */	dev_priv->dma_diff = ptr - reader;}static void via_pad_cache(drm_via_private_t * dev_priv, int qwords){	uint32_t *vb;	via_cmdbuf_wait(dev_priv, qwords + 2);	vb = via_get_dma(dev_priv);	VIA_OUT_RING_QW(HC_HEADER2, HC_ParaType_NotTex << 16);	via_align_buffer(dev_priv, vb, qwords);}static inline void via_dummy_bitblt(drm_via_private_t * dev_priv){	uint32_t *vb = via_get_dma(dev_priv);	SetReg2DAGP(0x0C, (0 | (0 << 16)));	SetReg2DAGP(0x10, 0 | (0 << 16));	SetReg2DAGP(0x0, 0x1 | 0x2000 | 0xAA000000);}static void via_cmdbuf_jump(drm_via_private_t * dev_priv){	uint32_t agp_base;	uint32_t pause_addr_lo, pause_addr_hi;	uint32_t jump_addr_lo, jump_addr_hi;	volatile uint32_t *last_pause_ptr;	agp_base = dev_priv->dma_offset + (uint32_t) dev_priv->agpAddr;	via_align_cmd(dev_priv, HC_HAGPBpID_JUMP, 0, &jump_addr_hi,		      &jump_addr_lo, 0);	dev_priv->dma_wrap = dev_priv->dma_low;	/*	 * Wrap command buffer to the beginning.	 */	dev_priv->dma_low = 0;	if (via_cmdbuf_wait(dev_priv, CMDBUF_ALIGNMENT_SIZE) != 0) {		DRM_ERROR("via_cmdbuf_jump failed\n");	}	via_dummy_bitblt(dev_priv);	via_dummy_bitblt(dev_priv);	last_pause_ptr =	    via_align_cmd(dev_priv, HC_HAGPBpID_PAUSE, 0, &pause_addr_hi,			  &pause_addr_lo, 0) - 1;	via_align_cmd(dev_priv, HC_HAGPBpID_PAUSE, 0, &pause_addr_hi,		      &pause_addr_lo, 0);	*last_pause_ptr = pause_addr_lo;	via_hook_segment( dev_priv, jump_addr_hi, jump_addr_lo, 0);}static void via_cmdbuf_rewind(drm_via_private_t * dev_priv){	via_cmdbuf_jump(dev_priv);}static void via_cmdbuf_flush(drm_via_private_t * dev_priv, uint32_t cmd_type){	uint32_t pause_addr_lo, pause_addr_hi;	via_align_cmd(dev_priv, cmd_type, 0, &pause_addr_hi, &pause_addr_lo, 0);	via_hook_segment(dev_priv, pause_addr_hi, pause_addr_lo, 0);}static void via_cmdbuf_pause(drm_via_private_t * dev_priv){	via_cmdbuf_flush(dev_priv, HC_HAGPBpID_PAUSE);}static void via_cmdbuf_reset(drm_via_private_t * dev_priv){	via_cmdbuf_flush(dev_priv, HC_HAGPBpID_STOP);	via_wait_idle(dev_priv);}/* * User interface to the space and lag functions. */static int via_cmdbuf_size(struct drm_device *dev, void *data, struct drm_file *file_priv){	drm_via_cmdbuf_size_t *d_siz = data;	int ret = 0;	uint32_t tmp_size, count;	drm_via_private_t *dev_priv;	DRM_DEBUG("via cmdbuf_size\n");	LOCK_TEST_WITH_RETURN(dev, file_priv);	dev_priv = (drm_via_private_t *) dev->dev_private;	if (dev_priv->ring.virtual_start == NULL) {		DRM_ERROR("%s called without initializing AGP ring buffer.\n",			  __FUNCTION__);		return -EFAULT;	}	count = 1000000;	tmp_size = d_siz->size;	switch (d_siz->func) {	case VIA_CMDBUF_SPACE:		while (((tmp_size = via_cmdbuf_space(dev_priv)) < d_siz->size)		       && count--) {			if (!d_siz->wait) {				break;			}		}		if (!count) {			DRM_ERROR("VIA_CMDBUF_SPACE timed out.\n");			ret = -EAGAIN;		}		break;	case VIA_CMDBUF_LAG:		while (((tmp_size = via_cmdbuf_lag(dev_priv)) > d_siz->size)		       && count--) {			if (!d_siz->wait) {				break;			}		}		if (!count) {			DRM_ERROR("VIA_CMDBUF_LAG timed out.\n");			ret = -EAGAIN;		}		break;	default:		ret = -EFAULT;	}	d_siz->size = tmp_size;	return ret;}struct drm_ioctl_desc via_ioctls[] = {	DRM_IOCTL_DEF(DRM_VIA_ALLOCMEM, via_mem_alloc, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_FREEMEM, via_mem_free, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_AGP_INIT, via_agp_init, DRM_AUTH|DRM_MASTER),	DRM_IOCTL_DEF(DRM_VIA_FB_INIT, via_fb_init, DRM_AUTH|DRM_MASTER),	DRM_IOCTL_DEF(DRM_VIA_MAP_INIT, via_map_init, DRM_AUTH|DRM_MASTER),	DRM_IOCTL_DEF(DRM_VIA_DEC_FUTEX, via_decoder_futex, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_DMA_INIT, via_dma_init, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_CMDBUFFER, via_cmdbuffer, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_FLUSH, via_flush_ioctl, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_PCICMD, via_pci_cmdbuffer, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_CMDBUF_SIZE, via_cmdbuf_size, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_WAIT_IRQ, via_wait_irq, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_DMA_BLIT, via_dma_blit, DRM_AUTH),	DRM_IOCTL_DEF(DRM_VIA_BLIT_SYNC, via_dma_blit_sync, DRM_AUTH)};int via_max_ioctl = DRM_ARRAY_SIZE(via_ioctls);

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?