xref: /openbmc/linux/drivers/gpu/drm/nouveau/dispnv04/hw.h (revision 3a2b9272)
11a646342SBen Skeggs /*
21a646342SBen Skeggs  * Copyright 2008 Stuart Bennett
31a646342SBen Skeggs  *
41a646342SBen Skeggs  * Permission is hereby granted, free of charge, to any person obtaining a
51a646342SBen Skeggs  * copy of this software and associated documentation files (the "Software"),
61a646342SBen Skeggs  * to deal in the Software without restriction, including without limitation
71a646342SBen Skeggs  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
81a646342SBen Skeggs  * and/or sell copies of the Software, and to permit persons to whom the
91a646342SBen Skeggs  * Software is furnished to do so, subject to the following conditions:
101a646342SBen Skeggs  *
111a646342SBen Skeggs  * The above copyright notice and this permission notice shall be included in
121a646342SBen Skeggs  * all copies or substantial portions of the Software.
131a646342SBen Skeggs  *
141a646342SBen Skeggs  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
151a646342SBen Skeggs  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
161a646342SBen Skeggs  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
171a646342SBen Skeggs  * THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
181a646342SBen Skeggs  * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
191a646342SBen Skeggs  * OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
201a646342SBen Skeggs  * SOFTWARE.
211a646342SBen Skeggs  */
221a646342SBen Skeggs 
231a646342SBen Skeggs #ifndef __NOUVEAU_HW_H__
241a646342SBen Skeggs #define __NOUVEAU_HW_H__
251a646342SBen Skeggs 
261a646342SBen Skeggs #include "disp.h"
271a646342SBen Skeggs #include "nvreg.h"
281a646342SBen Skeggs 
291a646342SBen Skeggs #include <subdev/bios/pll.h>
301a646342SBen Skeggs 
311a646342SBen Skeggs #define MASK(field) ( \
321a646342SBen Skeggs 	(0xffffffff >> (31 - ((1 ? field) - (0 ? field)))) << (0 ? field))
331a646342SBen Skeggs 
341a646342SBen Skeggs #define XLATE(src, srclowbit, outfield) ( \
351a646342SBen Skeggs 	(((src) >> (srclowbit)) << (0 ? outfield)) & MASK(outfield))
361a646342SBen Skeggs 
371a646342SBen Skeggs void NVWriteVgaSeq(struct drm_device *, int head, uint8_t index, uint8_t value);
381a646342SBen Skeggs uint8_t NVReadVgaSeq(struct drm_device *, int head, uint8_t index);
391a646342SBen Skeggs void NVWriteVgaGr(struct drm_device *, int head, uint8_t index, uint8_t value);
401a646342SBen Skeggs uint8_t NVReadVgaGr(struct drm_device *, int head, uint8_t index);
411a646342SBen Skeggs void NVSetOwner(struct drm_device *, int owner);
421a646342SBen Skeggs void NVBlankScreen(struct drm_device *, int head, bool blank);
431a646342SBen Skeggs int nouveau_hw_get_pllvals(struct drm_device *, enum nvbios_pll_type plltype,
44be83cd4eSBen Skeggs 			   struct nvkm_pll_vals *pllvals);
45be83cd4eSBen Skeggs int nouveau_hw_pllvals_to_clk(struct nvkm_pll_vals *pllvals);
461a646342SBen Skeggs int nouveau_hw_get_clock(struct drm_device *, enum nvbios_pll_type plltype);
471a646342SBen Skeggs void nouveau_hw_save_vga_fonts(struct drm_device *, bool save);
481a646342SBen Skeggs void nouveau_hw_save_state(struct drm_device *, int head,
491a646342SBen Skeggs 			   struct nv04_mode_state *state);
501a646342SBen Skeggs void nouveau_hw_load_state(struct drm_device *, int head,
511a646342SBen Skeggs 			   struct nv04_mode_state *state);
521a646342SBen Skeggs void nouveau_hw_load_state_palette(struct drm_device *, int head,
531a646342SBen Skeggs 				   struct nv04_mode_state *state);
541a646342SBen Skeggs 
551a646342SBen Skeggs /* nouveau_calc.c */
561a646342SBen Skeggs extern void nouveau_calc_arb(struct drm_device *, int vclk, int bpp,
571a646342SBen Skeggs 			     int *burst, int *lwm);
581a646342SBen Skeggs 
NVReadCRTC(struct drm_device * dev,int head,uint32_t reg)591a646342SBen Skeggs static inline uint32_t NVReadCRTC(struct drm_device *dev,
601a646342SBen Skeggs 					int head, uint32_t reg)
611a646342SBen Skeggs {
621167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
631a646342SBen Skeggs 	uint32_t val;
641a646342SBen Skeggs 	if (head)
651a646342SBen Skeggs 		reg += NV_PCRTC0_SIZE;
66db2bec18SBen Skeggs 	val = nvif_rd32(device, reg);
671a646342SBen Skeggs 	return val;
681a646342SBen Skeggs }
691a646342SBen Skeggs 
NVWriteCRTC(struct drm_device * dev,int head,uint32_t reg,uint32_t val)701a646342SBen Skeggs static inline void NVWriteCRTC(struct drm_device *dev,
711a646342SBen Skeggs 					int head, uint32_t reg, uint32_t val)
721a646342SBen Skeggs {
731167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
741a646342SBen Skeggs 	if (head)
751a646342SBen Skeggs 		reg += NV_PCRTC0_SIZE;
76db2bec18SBen Skeggs 	nvif_wr32(device, reg, val);
771a646342SBen Skeggs }
781a646342SBen Skeggs 
NVReadRAMDAC(struct drm_device * dev,int head,uint32_t reg)791a646342SBen Skeggs static inline uint32_t NVReadRAMDAC(struct drm_device *dev,
801a646342SBen Skeggs 					int head, uint32_t reg)
811a646342SBen Skeggs {
821167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
831a646342SBen Skeggs 	uint32_t val;
841a646342SBen Skeggs 	if (head)
851a646342SBen Skeggs 		reg += NV_PRAMDAC0_SIZE;
86db2bec18SBen Skeggs 	val = nvif_rd32(device, reg);
871a646342SBen Skeggs 	return val;
881a646342SBen Skeggs }
891a646342SBen Skeggs 
NVWriteRAMDAC(struct drm_device * dev,int head,uint32_t reg,uint32_t val)901a646342SBen Skeggs static inline void NVWriteRAMDAC(struct drm_device *dev,
911a646342SBen Skeggs 					int head, uint32_t reg, uint32_t val)
921a646342SBen Skeggs {
931167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
941a646342SBen Skeggs 	if (head)
951a646342SBen Skeggs 		reg += NV_PRAMDAC0_SIZE;
96db2bec18SBen Skeggs 	nvif_wr32(device, reg, val);
971a646342SBen Skeggs }
981a646342SBen Skeggs 
nv_read_tmds(struct drm_device * dev,int or,int dl,uint8_t address)991a646342SBen Skeggs static inline uint8_t nv_read_tmds(struct drm_device *dev,
1001a646342SBen Skeggs 					int or, int dl, uint8_t address)
1011a646342SBen Skeggs {
1021a646342SBen Skeggs 	int ramdac = (or & DCB_OUTPUT_C) >> 2;
1031a646342SBen Skeggs 
1041a646342SBen Skeggs 	NVWriteRAMDAC(dev, ramdac, NV_PRAMDAC_FP_TMDS_CONTROL + dl * 8,
1051a646342SBen Skeggs 	NV_PRAMDAC_FP_TMDS_CONTROL_WRITE_DISABLE | address);
1061a646342SBen Skeggs 	return NVReadRAMDAC(dev, ramdac, NV_PRAMDAC_FP_TMDS_DATA + dl * 8);
1071a646342SBen Skeggs }
1081a646342SBen Skeggs 
nv_write_tmds(struct drm_device * dev,int or,int dl,uint8_t address,uint8_t data)1091a646342SBen Skeggs static inline void nv_write_tmds(struct drm_device *dev,
1101a646342SBen Skeggs 					int or, int dl, uint8_t address,
1111a646342SBen Skeggs 					uint8_t data)
1121a646342SBen Skeggs {
1131a646342SBen Skeggs 	int ramdac = (or & DCB_OUTPUT_C) >> 2;
1141a646342SBen Skeggs 
1151a646342SBen Skeggs 	NVWriteRAMDAC(dev, ramdac, NV_PRAMDAC_FP_TMDS_DATA + dl * 8, data);
1161a646342SBen Skeggs 	NVWriteRAMDAC(dev, ramdac, NV_PRAMDAC_FP_TMDS_CONTROL + dl * 8, address);
1171a646342SBen Skeggs }
1181a646342SBen Skeggs 
NVWriteVgaCrtc(struct drm_device * dev,int head,uint8_t index,uint8_t value)1191a646342SBen Skeggs static inline void NVWriteVgaCrtc(struct drm_device *dev,
1201a646342SBen Skeggs 					int head, uint8_t index, uint8_t value)
1211a646342SBen Skeggs {
1221167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
123db2bec18SBen Skeggs 	nvif_wr08(device, NV_PRMCIO_CRX__COLOR + head * NV_PRMCIO_SIZE, index);
124db2bec18SBen Skeggs 	nvif_wr08(device, NV_PRMCIO_CR__COLOR + head * NV_PRMCIO_SIZE, value);
1251a646342SBen Skeggs }
1261a646342SBen Skeggs 
NVReadVgaCrtc(struct drm_device * dev,int head,uint8_t index)1271a646342SBen Skeggs static inline uint8_t NVReadVgaCrtc(struct drm_device *dev,
1281a646342SBen Skeggs 					int head, uint8_t index)
1291a646342SBen Skeggs {
1301167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
1311a646342SBen Skeggs 	uint8_t val;
132db2bec18SBen Skeggs 	nvif_wr08(device, NV_PRMCIO_CRX__COLOR + head * NV_PRMCIO_SIZE, index);
133db2bec18SBen Skeggs 	val = nvif_rd08(device, NV_PRMCIO_CR__COLOR + head * NV_PRMCIO_SIZE);
1341a646342SBen Skeggs 	return val;
1351a646342SBen Skeggs }
1361a646342SBen Skeggs 
1371a646342SBen Skeggs /* CR57 and CR58 are a fun pair of regs. CR57 provides an index (0-0xf) for CR58
1381a646342SBen Skeggs  * I suspect they in fact do nothing, but are merely a way to carry useful
1391a646342SBen Skeggs  * per-head variables around
1401a646342SBen Skeggs  *
1411a646342SBen Skeggs  * Known uses:
1421a646342SBen Skeggs  * CR57		CR58
1431a646342SBen Skeggs  * 0x00		index to the appropriate dcb entry (or 7f for inactive)
1441a646342SBen Skeggs  * 0x02		dcb entry's "or" value (or 00 for inactive)
1451a646342SBen Skeggs  * 0x03		bit0 set for dual link (LVDS, possibly elsewhere too)
1461a646342SBen Skeggs  * 0x08 or 0x09	pxclk in MHz
1471a646342SBen Skeggs  * 0x0f		laptop panel info -	low nibble for PEXTDEV_BOOT_0 strap
1481a646342SBen Skeggs  * 					high nibble for xlat strap value
1491a646342SBen Skeggs  */
1501a646342SBen Skeggs 
1511a646342SBen Skeggs static inline void
NVWriteVgaCrtc5758(struct drm_device * dev,int head,uint8_t index,uint8_t value)1521a646342SBen Skeggs NVWriteVgaCrtc5758(struct drm_device *dev, int head, uint8_t index, uint8_t value)
1531a646342SBen Skeggs {
1541a646342SBen Skeggs 	NVWriteVgaCrtc(dev, head, NV_CIO_CRE_57, index);
1551a646342SBen Skeggs 	NVWriteVgaCrtc(dev, head, NV_CIO_CRE_58, value);
1561a646342SBen Skeggs }
1571a646342SBen Skeggs 
NVReadVgaCrtc5758(struct drm_device * dev,int head,uint8_t index)1581a646342SBen Skeggs static inline uint8_t NVReadVgaCrtc5758(struct drm_device *dev, int head, uint8_t index)
1591a646342SBen Skeggs {
1601a646342SBen Skeggs 	NVWriteVgaCrtc(dev, head, NV_CIO_CRE_57, index);
1611a646342SBen Skeggs 	return NVReadVgaCrtc(dev, head, NV_CIO_CRE_58);
1621a646342SBen Skeggs }
1631a646342SBen Skeggs 
NVReadPRMVIO(struct drm_device * dev,int head,uint32_t reg)1641a646342SBen Skeggs static inline uint8_t NVReadPRMVIO(struct drm_device *dev,
1651a646342SBen Skeggs 					int head, uint32_t reg)
1661a646342SBen Skeggs {
1671167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
1681a646342SBen Skeggs 	struct nouveau_drm *drm = nouveau_drm(dev);
1691a646342SBen Skeggs 	uint8_t val;
1701a646342SBen Skeggs 
1711a646342SBen Skeggs 	/* Only NV4x have two pvio ranges; other twoHeads cards MUST call
1721a646342SBen Skeggs 	 * NVSetOwner for the relevant head to be programmed */
1731167c6bcSBen Skeggs 	if (head && drm->client.device.info.family == NV_DEVICE_INFO_V0_CURIE)
1741a646342SBen Skeggs 		reg += NV_PRMVIO_SIZE;
1751a646342SBen Skeggs 
176db2bec18SBen Skeggs 	val = nvif_rd08(device, reg);
1771a646342SBen Skeggs 	return val;
1781a646342SBen Skeggs }
1791a646342SBen Skeggs 
NVWritePRMVIO(struct drm_device * dev,int head,uint32_t reg,uint8_t value)1801a646342SBen Skeggs static inline void NVWritePRMVIO(struct drm_device *dev,
1811a646342SBen Skeggs 					int head, uint32_t reg, uint8_t value)
1821a646342SBen Skeggs {
1831167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
1841a646342SBen Skeggs 	struct nouveau_drm *drm = nouveau_drm(dev);
1851a646342SBen Skeggs 
1861a646342SBen Skeggs 	/* Only NV4x have two pvio ranges; other twoHeads cards MUST call
1871a646342SBen Skeggs 	 * NVSetOwner for the relevant head to be programmed */
1881167c6bcSBen Skeggs 	if (head && drm->client.device.info.family == NV_DEVICE_INFO_V0_CURIE)
1891a646342SBen Skeggs 		reg += NV_PRMVIO_SIZE;
1901a646342SBen Skeggs 
191db2bec18SBen Skeggs 	nvif_wr08(device, reg, value);
1921a646342SBen Skeggs }
1931a646342SBen Skeggs 
NVSetEnablePalette(struct drm_device * dev,int head,bool enable)1941a646342SBen Skeggs static inline void NVSetEnablePalette(struct drm_device *dev, int head, bool enable)
1951a646342SBen Skeggs {
1961167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
197db2bec18SBen Skeggs 	nvif_rd08(device, NV_PRMCIO_INP0__COLOR + head * NV_PRMCIO_SIZE);
198db2bec18SBen Skeggs 	nvif_wr08(device, NV_PRMCIO_ARX + head * NV_PRMCIO_SIZE, enable ? 0 : 0x20);
1991a646342SBen Skeggs }
2001a646342SBen Skeggs 
NVGetEnablePalette(struct drm_device * dev,int head)2011a646342SBen Skeggs static inline bool NVGetEnablePalette(struct drm_device *dev, int head)
2021a646342SBen Skeggs {
2031167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
204db2bec18SBen Skeggs 	nvif_rd08(device, NV_PRMCIO_INP0__COLOR + head * NV_PRMCIO_SIZE);
205db2bec18SBen Skeggs 	return !(nvif_rd08(device, NV_PRMCIO_ARX + head * NV_PRMCIO_SIZE) & 0x20);
2061a646342SBen Skeggs }
2071a646342SBen Skeggs 
NVWriteVgaAttr(struct drm_device * dev,int head,uint8_t index,uint8_t value)2081a646342SBen Skeggs static inline void NVWriteVgaAttr(struct drm_device *dev,
2091a646342SBen Skeggs 					int head, uint8_t index, uint8_t value)
2101a646342SBen Skeggs {
2111167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
2121a646342SBen Skeggs 	if (NVGetEnablePalette(dev, head))
2131a646342SBen Skeggs 		index &= ~0x20;
2141a646342SBen Skeggs 	else
2151a646342SBen Skeggs 		index |= 0x20;
2161a646342SBen Skeggs 
217db2bec18SBen Skeggs 	nvif_rd08(device, NV_PRMCIO_INP0__COLOR + head * NV_PRMCIO_SIZE);
218db2bec18SBen Skeggs 	nvif_wr08(device, NV_PRMCIO_ARX + head * NV_PRMCIO_SIZE, index);
219db2bec18SBen Skeggs 	nvif_wr08(device, NV_PRMCIO_AR__WRITE + head * NV_PRMCIO_SIZE, value);
2201a646342SBen Skeggs }
2211a646342SBen Skeggs 
NVReadVgaAttr(struct drm_device * dev,int head,uint8_t index)2221a646342SBen Skeggs static inline uint8_t NVReadVgaAttr(struct drm_device *dev,
2231a646342SBen Skeggs 					int head, uint8_t index)
2241a646342SBen Skeggs {
2251167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
2261a646342SBen Skeggs 	uint8_t val;
2271a646342SBen Skeggs 	if (NVGetEnablePalette(dev, head))
2281a646342SBen Skeggs 		index &= ~0x20;
2291a646342SBen Skeggs 	else
2301a646342SBen Skeggs 		index |= 0x20;
2311a646342SBen Skeggs 
232db2bec18SBen Skeggs 	nvif_rd08(device, NV_PRMCIO_INP0__COLOR + head * NV_PRMCIO_SIZE);
233db2bec18SBen Skeggs 	nvif_wr08(device, NV_PRMCIO_ARX + head * NV_PRMCIO_SIZE, index);
234db2bec18SBen Skeggs 	val = nvif_rd08(device, NV_PRMCIO_AR__READ + head * NV_PRMCIO_SIZE);
2351a646342SBen Skeggs 	return val;
2361a646342SBen Skeggs }
2371a646342SBen Skeggs 
NVVgaSeqReset(struct drm_device * dev,int head,bool start)2381a646342SBen Skeggs static inline void NVVgaSeqReset(struct drm_device *dev, int head, bool start)
2391a646342SBen Skeggs {
2401a646342SBen Skeggs 	NVWriteVgaSeq(dev, head, NV_VIO_SR_RESET_INDEX, start ? 0x1 : 0x3);
2411a646342SBen Skeggs }
2421a646342SBen Skeggs 
NVVgaProtect(struct drm_device * dev,int head,bool protect)2431a646342SBen Skeggs static inline void NVVgaProtect(struct drm_device *dev, int head, bool protect)
2441a646342SBen Skeggs {
2451a646342SBen Skeggs 	uint8_t seq1 = NVReadVgaSeq(dev, head, NV_VIO_SR_CLOCK_INDEX);
2461a646342SBen Skeggs 
2471a646342SBen Skeggs 	if (protect) {
2481a646342SBen Skeggs 		NVVgaSeqReset(dev, head, true);
2491a646342SBen Skeggs 		NVWriteVgaSeq(dev, head, NV_VIO_SR_CLOCK_INDEX, seq1 | 0x20);
2501a646342SBen Skeggs 	} else {
2511a646342SBen Skeggs 		/* Reenable sequencer, then turn on screen */
2521a646342SBen Skeggs 		NVWriteVgaSeq(dev, head, NV_VIO_SR_CLOCK_INDEX, seq1 & ~0x20);   /* reenable display */
2531a646342SBen Skeggs 		NVVgaSeqReset(dev, head, false);
2541a646342SBen Skeggs 	}
2551a646342SBen Skeggs 	NVSetEnablePalette(dev, head, protect);
2561a646342SBen Skeggs }
2571a646342SBen Skeggs 
2581a646342SBen Skeggs static inline bool
nv_heads_tied(struct drm_device * dev)2591a646342SBen Skeggs nv_heads_tied(struct drm_device *dev)
2601a646342SBen Skeggs {
2611167c6bcSBen Skeggs 	struct nvif_object *device = &nouveau_drm(dev)->client.device.object;
2621a646342SBen Skeggs 	struct nouveau_drm *drm = nouveau_drm(dev);
2631a646342SBen Skeggs 
2641167c6bcSBen Skeggs 	if (drm->client.device.info.chipset == 0x11)
265db2bec18SBen Skeggs 		return !!(nvif_rd32(device, NV_PBUS_DEBUG_1) & (1 << 28));
2661a646342SBen Skeggs 
2671a646342SBen Skeggs 	return NVReadVgaCrtc(dev, 0, NV_CIO_CRE_44) & 0x4;
2681a646342SBen Skeggs }
2691a646342SBen Skeggs 
2701a646342SBen Skeggs /* makes cr0-7 on the specified head read-only */
2711a646342SBen Skeggs static inline bool
nv_lock_vga_crtc_base(struct drm_device * dev,int head,bool lock)2721a646342SBen Skeggs nv_lock_vga_crtc_base(struct drm_device *dev, int head, bool lock)
2731a646342SBen Skeggs {
2741a646342SBen Skeggs 	uint8_t cr11 = NVReadVgaCrtc(dev, head, NV_CIO_CR_VRE_INDEX);
2751a646342SBen Skeggs 	bool waslocked = cr11 & 0x80;
2761a646342SBen Skeggs 
2771a646342SBen Skeggs 	if (lock)
2781a646342SBen Skeggs 		cr11 |= 0x80;
2791a646342SBen Skeggs 	else
2801a646342SBen Skeggs 		cr11 &= ~0x80;
2811a646342SBen Skeggs 	NVWriteVgaCrtc(dev, head, NV_CIO_CR_VRE_INDEX, cr11);
2821a646342SBen Skeggs 
2831a646342SBen Skeggs 	return waslocked;
2841a646342SBen Skeggs }
2851a646342SBen Skeggs 
2861a646342SBen Skeggs static inline void
nv_lock_vga_crtc_shadow(struct drm_device * dev,int head,int lock)2871a646342SBen Skeggs nv_lock_vga_crtc_shadow(struct drm_device *dev, int head, int lock)
2881a646342SBen Skeggs {
2891a646342SBen Skeggs 	/* shadow lock: connects 0x60?3d? regs to "real" 0x3d? regs
2901a646342SBen Skeggs 	 * bit7: unlocks HDT, HBS, HBE, HRS, HRE, HEB
2911a646342SBen Skeggs 	 * bit6: seems to have some effect on CR09 (double scan, VBS_9)
2921a646342SBen Skeggs 	 * bit5: unlocks HDE
2931a646342SBen Skeggs 	 * bit4: unlocks VDE
2941a646342SBen Skeggs 	 * bit3: unlocks VDT, OVL, VRS, ?VRE?, VBS, VBE, LSR, EBR
2951a646342SBen Skeggs 	 * bit2: same as bit 1 of 0x60?804
2961a646342SBen Skeggs 	 * bit0: same as bit 0 of 0x60?804
2971a646342SBen Skeggs 	 */
2981a646342SBen Skeggs 
2991a646342SBen Skeggs 	uint8_t cr21 = lock;
3001a646342SBen Skeggs 
3011a646342SBen Skeggs 	if (lock < 0)
3021a646342SBen Skeggs 		/* 0xfa is generic "unlock all" mask */
3031a646342SBen Skeggs 		cr21 = NVReadVgaCrtc(dev, head, NV_CIO_CRE_21) | 0xfa;
3041a646342SBen Skeggs 
3051a646342SBen Skeggs 	NVWriteVgaCrtc(dev, head, NV_CIO_CRE_21, cr21);
3061a646342SBen Skeggs }
3071a646342SBen Skeggs 
3081a646342SBen Skeggs /* renders the extended crtc regs (cr19+) on all crtcs impervious:
3091a646342SBen Skeggs  * immutable and unreadable
3101a646342SBen Skeggs  */
3111a646342SBen Skeggs static inline bool
NVLockVgaCrtcs(struct drm_device * dev,bool lock)3121a646342SBen Skeggs NVLockVgaCrtcs(struct drm_device *dev, bool lock)
3131a646342SBen Skeggs {
3141a646342SBen Skeggs 	struct nouveau_drm *drm = nouveau_drm(dev);
3151a646342SBen Skeggs 	bool waslocked = !NVReadVgaCrtc(dev, 0, NV_CIO_SR_LOCK_INDEX);
3161a646342SBen Skeggs 
3171a646342SBen Skeggs 	NVWriteVgaCrtc(dev, 0, NV_CIO_SR_LOCK_INDEX,
3181a646342SBen Skeggs 		       lock ? NV_CIO_SR_LOCK_VALUE : NV_CIO_SR_UNLOCK_RW_VALUE);
3191a646342SBen Skeggs 	/* NV11 has independently lockable extended crtcs, except when tied */
3201167c6bcSBen Skeggs 	if (drm->client.device.info.chipset == 0x11 && !nv_heads_tied(dev))
3211a646342SBen Skeggs 		NVWriteVgaCrtc(dev, 1, NV_CIO_SR_LOCK_INDEX,
3221a646342SBen Skeggs 			       lock ? NV_CIO_SR_LOCK_VALUE :
3231a646342SBen Skeggs 				      NV_CIO_SR_UNLOCK_RW_VALUE);
3241a646342SBen Skeggs 
3251a646342SBen Skeggs 	return waslocked;
3261a646342SBen Skeggs }
3271a646342SBen Skeggs 
3281a646342SBen Skeggs /* nv04 cursor max dimensions of 32x32 (A1R5G5B5) */
3291a646342SBen Skeggs #define NV04_CURSOR_SIZE 32
3301a646342SBen Skeggs /* limit nv10 cursors to 64x64 (ARGB8) (we could go to 64x255) */
3311a646342SBen Skeggs #define NV10_CURSOR_SIZE 64
3321a646342SBen Skeggs 
nv_cursor_width(struct drm_device * dev)3331a646342SBen Skeggs static inline int nv_cursor_width(struct drm_device *dev)
3341a646342SBen Skeggs {
3351a646342SBen Skeggs 	struct nouveau_drm *drm = nouveau_drm(dev);
3361a646342SBen Skeggs 
3371167c6bcSBen Skeggs 	return drm->client.device.info.family >= NV_DEVICE_INFO_V0_CELSIUS ? NV10_CURSOR_SIZE : NV04_CURSOR_SIZE;
3381a646342SBen Skeggs }
3391a646342SBen Skeggs 
3401a646342SBen Skeggs static inline void
nv_fix_nv40_hw_cursor(struct drm_device * dev,int head)3411a646342SBen Skeggs nv_fix_nv40_hw_cursor(struct drm_device *dev, int head)
3421a646342SBen Skeggs {
3431a646342SBen Skeggs 	/* on some nv40 (such as the "true" (in the NV_PFB_BOOT_0 sense) nv40,
3441a646342SBen Skeggs 	 * the gf6800gt) a hardware bug requires a write to PRAMDAC_CURSOR_POS
3451a646342SBen Skeggs 	 * for changes to the CRTC CURCTL regs to take effect, whether changing
3461a646342SBen Skeggs 	 * the pixmap location, or just showing/hiding the cursor
3471a646342SBen Skeggs 	 */
3481a646342SBen Skeggs 	uint32_t curpos = NVReadRAMDAC(dev, head, NV_PRAMDAC_CU_START_POS);
3491a646342SBen Skeggs 	NVWriteRAMDAC(dev, head, NV_PRAMDAC_CU_START_POS, curpos);
3501a646342SBen Skeggs }
3511a646342SBen Skeggs 
3521a646342SBen Skeggs static inline void
nv_set_crtc_base(struct drm_device * dev,int head,uint32_t offset)3531a646342SBen Skeggs nv_set_crtc_base(struct drm_device *dev, int head, uint32_t offset)
3541a646342SBen Skeggs {
3551a646342SBen Skeggs 	struct nouveau_drm *drm = nouveau_drm(dev);
3561a646342SBen Skeggs 
3571a646342SBen Skeggs 	NVWriteCRTC(dev, head, NV_PCRTC_START, offset);
3581a646342SBen Skeggs 
3591167c6bcSBen Skeggs 	if (drm->client.device.info.family == NV_DEVICE_INFO_V0_TNT) {
3601a646342SBen Skeggs 		/*
3611a646342SBen Skeggs 		 * Hilarious, the 24th bit doesn't want to stick to
3621a646342SBen Skeggs 		 * PCRTC_START...
3631a646342SBen Skeggs 		 */
3641a646342SBen Skeggs 		int cre_heb = NVReadVgaCrtc(dev, head, NV_CIO_CRE_HEB__INDEX);
3651a646342SBen Skeggs 
3661a646342SBen Skeggs 		NVWriteVgaCrtc(dev, head, NV_CIO_CRE_HEB__INDEX,
3671a646342SBen Skeggs 			       (cre_heb & ~0x40) | ((offset >> 18) & 0x40));
3681a646342SBen Skeggs 	}
3691a646342SBen Skeggs }
3701a646342SBen Skeggs 
3711a646342SBen Skeggs static inline void
nv_show_cursor(struct drm_device * dev,int head,bool show)3721a646342SBen Skeggs nv_show_cursor(struct drm_device *dev, int head, bool show)
3731a646342SBen Skeggs {
3741a646342SBen Skeggs 	struct nouveau_drm *drm = nouveau_drm(dev);
3751a646342SBen Skeggs 	uint8_t *curctl1 =
3761a646342SBen Skeggs 		&nv04_display(dev)->mode_reg.crtc_reg[head].CRTC[NV_CIO_CRE_HCUR_ADDR1_INDEX];
3771a646342SBen Skeggs 
3781a646342SBen Skeggs 	if (show)
3791a646342SBen Skeggs 		*curctl1 |= MASK(NV_CIO_CRE_HCUR_ADDR1_ENABLE);
3801a646342SBen Skeggs 	else
3811a646342SBen Skeggs 		*curctl1 &= ~MASK(NV_CIO_CRE_HCUR_ADDR1_ENABLE);
3821a646342SBen Skeggs 	NVWriteVgaCrtc(dev, head, NV_CIO_CRE_HCUR_ADDR1_INDEX, *curctl1);
3831a646342SBen Skeggs 
3841167c6bcSBen Skeggs 	if (drm->client.device.info.family == NV_DEVICE_INFO_V0_CURIE)
3851a646342SBen Skeggs 		nv_fix_nv40_hw_cursor(dev, head);
3861a646342SBen Skeggs }
3871a646342SBen Skeggs 
3881a646342SBen Skeggs static inline uint32_t
nv_pitch_align(struct drm_device * dev,uint32_t width,int bpp)3891a646342SBen Skeggs nv_pitch_align(struct drm_device *dev, uint32_t width, int bpp)
3901a646342SBen Skeggs {
3911a646342SBen Skeggs 	struct nouveau_drm *drm = nouveau_drm(dev);
3921a646342SBen Skeggs 	int mask;
3931a646342SBen Skeggs 
3941a646342SBen Skeggs 	if (bpp == 15)
3951a646342SBen Skeggs 		bpp = 16;
3961a646342SBen Skeggs 	if (bpp == 24)
3971a646342SBen Skeggs 		bpp = 8;
3981a646342SBen Skeggs 
3991a646342SBen Skeggs 	/* Alignment requirements taken from the Haiku driver */
4001167c6bcSBen Skeggs 	if (drm->client.device.info.family == NV_DEVICE_INFO_V0_TNT)
4011a646342SBen Skeggs 		mask = 128 / bpp - 1;
4021a646342SBen Skeggs 	else
4031a646342SBen Skeggs 		mask = 512 / bpp - 1;
4041a646342SBen Skeggs 
4051a646342SBen Skeggs 	return (width + mask) & ~mask;
4061a646342SBen Skeggs }
4071a646342SBen Skeggs 
4081a646342SBen Skeggs #endif	/* __NOUVEAU_HW_H__ */
409