⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 via_dma.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 2 页
字号:
		return DRM_ERR(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;}int via_pci_cmdbuffer(DRM_IOCTL_ARGS){	DRM_DEVICE;	drm_via_cmdbuffer_t cmdbuf;	int ret;	LOCK_TEST_WITH_RETURN(dev, filp);	DRM_COPY_FROM_USER_IOCTL(cmdbuf, (drm_via_cmdbuffer_t __user *) data,				 sizeof(cmdbuf));	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;	via_flush_write_combine();	while (!*(via_get_dma(dev_priv) - 1)) ;	*dev_priv->last_pause_ptr = pause_addr_lo;	via_flush_write_combine();	/*	 * The below statement is inserted to really force the flush.	 * Not sure it is needed.	 */	while (!*dev_priv->last_pause_ptr) ;	dev_priv->last_pause_ptr = via_get_dma(dev_priv) - 1;	while (!*dev_priv->last_pause_ptr) ;	paused = 0;	count = 20;	while (!(paused = (VIA_READ(0x41c) & 0x80000000)) && count--) ;	if ((count <= 8) && (count >= 0)) {		uint32_t rgtr, ptr;		rgtr = *(dev_priv->hw_addr_ptr);		ptr = ((char *)dev_priv->last_pause_ptr - dev_priv->dma_ptr) +		    dev_priv->dma_offset + (uint32_t) dev_priv->agpAddr + 4 -		    CMDBUF_ALIGNMENT_SIZE;		if (rgtr <= ptr) {			DRM_ERROR			    ("Command regulator\npaused at count %d, address %x, "			     "while current pause address is %x.\n"			     "Please mail this message to "			     "<unichrome-devel@lists.sourceforge.net>\n", count,			     rgtr, ptr);		}	}	if (paused && !no_pci_fire) {		uint32_t rgtr, ptr;		uint32_t ptr_low;		count = 1000000;		while ((VIA_READ(VIA_REG_STATUS) & VIA_CMD_RGTR_BUSY)		       && count--) ;		rgtr = *(dev_priv->hw_addr_ptr);		ptr = ((char *)paused_at - dev_priv->dma_ptr) +		    dev_priv->dma_offset + (uint32_t) dev_priv->agpAddr + 4;		ptr_low = (ptr > 3 * CMDBUF_ALIGNMENT_SIZE) ?		    ptr - 3 * CMDBUF_ALIGNMENT_SIZE : 0;		if (rgtr <= ptr && rgtr >= ptr_low) {			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);		}	}	return paused;}static int via_wait_idle(drm_via_private_t * dev_priv){	int count = 10000000;	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;	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();	while (!*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);	VIA_WRITE(VIA_REG_TRANSPACE, command | HC_HAGPCMNT_MASK);}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;	uint32_t dma_low_save1, dma_low_save2;	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;	dma_low_save1 = dev_priv->dma_low;	/*	 * Now, set a trap that will pause the regulator if it tries to rerun the old	 * command buffer. (Which may happen if via_hook_segment detecs a command regulator pause	 * and reissues the jump command over PCI, while the regulator has already taken the jump	 * and actually paused at the current buffer end).	 * There appears to be no other way to detect this condition, since the hw_addr_pointer	 * does not seem to get updated immediately when a jump occurs.	 */	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;	dma_low_save2 = dev_priv->dma_low;	dev_priv->dma_low = dma_low_save1;	via_hook_segment(dev_priv, jump_addr_hi, jump_addr_lo, 0);	dev_priv->dma_low = dma_low_save2;	via_hook_segment(dev_priv, pause_addr_hi, pause_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. */int via_cmdbuf_size(DRM_IOCTL_ARGS){	DRM_DEVICE;	drm_via_cmdbuf_size_t d_siz;	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, filp);	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 DRM_ERR(EFAULT);	}	DRM_COPY_FROM_USER_IOCTL(d_siz, (drm_via_cmdbuf_size_t __user *) data,				 sizeof(d_siz));	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 = DRM_ERR(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 = DRM_ERR(EAGAIN);		}		break;	default:		ret = DRM_ERR(EFAULT);	}	d_siz.size = tmp_size;	DRM_COPY_TO_USER_IOCTL((drm_via_cmdbuf_size_t __user *) data, d_siz,			       sizeof(d_siz));	return ret;}

⌨️ 快捷键说明

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