18c2ecf20Sopenharmony_ci/*
28c2ecf20Sopenharmony_ci * Copyright 2015 Red Hat Inc.
38c2ecf20Sopenharmony_ci *
48c2ecf20Sopenharmony_ci * Permission is hereby granted, free of charge, to any person obtaining a
58c2ecf20Sopenharmony_ci * copy of this software and associated documentation files (the "Software"),
68c2ecf20Sopenharmony_ci * to deal in the Software without restriction, including without limitation
78c2ecf20Sopenharmony_ci * the rights to use, copy, modify, merge, publish, distribute, sublicense,
88c2ecf20Sopenharmony_ci * and/or sell copies of the Software, and to permit persons to whom the
98c2ecf20Sopenharmony_ci * Software is furnished to do so, subject to the following conditions:
108c2ecf20Sopenharmony_ci *
118c2ecf20Sopenharmony_ci * The above copyright notice and this permission notice shall be included in
128c2ecf20Sopenharmony_ci * all copies or substantial portions of the Software.
138c2ecf20Sopenharmony_ci *
148c2ecf20Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
158c2ecf20Sopenharmony_ci * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
168c2ecf20Sopenharmony_ci * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
178c2ecf20Sopenharmony_ci * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
188c2ecf20Sopenharmony_ci * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
198c2ecf20Sopenharmony_ci * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
208c2ecf20Sopenharmony_ci * OTHER DEALINGS IN THE SOFTWARE.
218c2ecf20Sopenharmony_ci *
228c2ecf20Sopenharmony_ci * Authors: Dave Airlie
238c2ecf20Sopenharmony_ci */
248c2ecf20Sopenharmony_ci
258c2ecf20Sopenharmony_ci#include <drm/radeon_drm.h>
268c2ecf20Sopenharmony_ci#include "radeon.h"
278c2ecf20Sopenharmony_ci#include "nid.h"
288c2ecf20Sopenharmony_ci
298c2ecf20Sopenharmony_ci#define AUX_RX_ERROR_FLAGS (AUX_SW_RX_OVERFLOW |	     \
308c2ecf20Sopenharmony_ci			    AUX_SW_RX_HPD_DISCON |	     \
318c2ecf20Sopenharmony_ci			    AUX_SW_RX_PARTIAL_BYTE |	     \
328c2ecf20Sopenharmony_ci			    AUX_SW_NON_AUX_MODE |	     \
338c2ecf20Sopenharmony_ci			    AUX_SW_RX_SYNC_INVALID_L |	     \
348c2ecf20Sopenharmony_ci			    AUX_SW_RX_SYNC_INVALID_H |	     \
358c2ecf20Sopenharmony_ci			    AUX_SW_RX_INVALID_START |	     \
368c2ecf20Sopenharmony_ci			    AUX_SW_RX_RECV_NO_DET |	     \
378c2ecf20Sopenharmony_ci			    AUX_SW_RX_RECV_INVALID_H |	     \
388c2ecf20Sopenharmony_ci			    AUX_SW_RX_RECV_INVALID_V)
398c2ecf20Sopenharmony_ci
408c2ecf20Sopenharmony_ci#define AUX_SW_REPLY_GET_BYTE_COUNT(x) (((x) >> 24) & 0x1f)
418c2ecf20Sopenharmony_ci
428c2ecf20Sopenharmony_ci#define BARE_ADDRESS_SIZE 3
438c2ecf20Sopenharmony_ci
448c2ecf20Sopenharmony_cistatic const u32 aux_offset[] =
458c2ecf20Sopenharmony_ci{
468c2ecf20Sopenharmony_ci	0x6200 - 0x6200,
478c2ecf20Sopenharmony_ci	0x6250 - 0x6200,
488c2ecf20Sopenharmony_ci	0x62a0 - 0x6200,
498c2ecf20Sopenharmony_ci	0x6300 - 0x6200,
508c2ecf20Sopenharmony_ci	0x6350 - 0x6200,
518c2ecf20Sopenharmony_ci	0x63a0 - 0x6200,
528c2ecf20Sopenharmony_ci};
538c2ecf20Sopenharmony_ci
548c2ecf20Sopenharmony_cissize_t
558c2ecf20Sopenharmony_ciradeon_dp_aux_transfer_native(struct drm_dp_aux *aux, struct drm_dp_aux_msg *msg)
568c2ecf20Sopenharmony_ci{
578c2ecf20Sopenharmony_ci	struct radeon_i2c_chan *chan =
588c2ecf20Sopenharmony_ci		container_of(aux, struct radeon_i2c_chan, aux);
598c2ecf20Sopenharmony_ci	struct drm_device *dev = chan->dev;
608c2ecf20Sopenharmony_ci	struct radeon_device *rdev = dev->dev_private;
618c2ecf20Sopenharmony_ci	int ret = 0, i;
628c2ecf20Sopenharmony_ci	uint32_t tmp, ack = 0;
638c2ecf20Sopenharmony_ci	int instance = chan->rec.i2c_id & 0xf;
648c2ecf20Sopenharmony_ci	u8 byte;
658c2ecf20Sopenharmony_ci	u8 *buf = msg->buffer;
668c2ecf20Sopenharmony_ci	int retry_count = 0;
678c2ecf20Sopenharmony_ci	int bytes;
688c2ecf20Sopenharmony_ci	int msize;
698c2ecf20Sopenharmony_ci	bool is_write = false;
708c2ecf20Sopenharmony_ci
718c2ecf20Sopenharmony_ci	if (WARN_ON(msg->size > 16))
728c2ecf20Sopenharmony_ci		return -E2BIG;
738c2ecf20Sopenharmony_ci
748c2ecf20Sopenharmony_ci	switch (msg->request & ~DP_AUX_I2C_MOT) {
758c2ecf20Sopenharmony_ci	case DP_AUX_NATIVE_WRITE:
768c2ecf20Sopenharmony_ci	case DP_AUX_I2C_WRITE:
778c2ecf20Sopenharmony_ci		is_write = true;
788c2ecf20Sopenharmony_ci		break;
798c2ecf20Sopenharmony_ci	case DP_AUX_NATIVE_READ:
808c2ecf20Sopenharmony_ci	case DP_AUX_I2C_READ:
818c2ecf20Sopenharmony_ci		break;
828c2ecf20Sopenharmony_ci	default:
838c2ecf20Sopenharmony_ci		return -EINVAL;
848c2ecf20Sopenharmony_ci	}
858c2ecf20Sopenharmony_ci
868c2ecf20Sopenharmony_ci	/* work out two sizes required */
878c2ecf20Sopenharmony_ci	msize = 0;
888c2ecf20Sopenharmony_ci	bytes = BARE_ADDRESS_SIZE;
898c2ecf20Sopenharmony_ci	if (msg->size) {
908c2ecf20Sopenharmony_ci		msize = msg->size - 1;
918c2ecf20Sopenharmony_ci		bytes++;
928c2ecf20Sopenharmony_ci		if (is_write)
938c2ecf20Sopenharmony_ci			bytes += msg->size;
948c2ecf20Sopenharmony_ci	}
958c2ecf20Sopenharmony_ci
968c2ecf20Sopenharmony_ci	mutex_lock(&chan->mutex);
978c2ecf20Sopenharmony_ci
988c2ecf20Sopenharmony_ci	/* switch the pad to aux mode */
998c2ecf20Sopenharmony_ci	tmp = RREG32(chan->rec.mask_clk_reg);
1008c2ecf20Sopenharmony_ci	tmp |= (1 << 16);
1018c2ecf20Sopenharmony_ci	WREG32(chan->rec.mask_clk_reg, tmp);
1028c2ecf20Sopenharmony_ci
1038c2ecf20Sopenharmony_ci	/* setup AUX control register with correct HPD pin */
1048c2ecf20Sopenharmony_ci	tmp = RREG32(AUX_CONTROL + aux_offset[instance]);
1058c2ecf20Sopenharmony_ci
1068c2ecf20Sopenharmony_ci	tmp &= AUX_HPD_SEL(0x7);
1078c2ecf20Sopenharmony_ci	tmp |= AUX_HPD_SEL(chan->rec.hpd);
1088c2ecf20Sopenharmony_ci	tmp |= AUX_EN | AUX_LS_READ_EN;
1098c2ecf20Sopenharmony_ci
1108c2ecf20Sopenharmony_ci	WREG32(AUX_CONTROL + aux_offset[instance], tmp);
1118c2ecf20Sopenharmony_ci
1128c2ecf20Sopenharmony_ci	/* atombios appears to write this twice lets copy it */
1138c2ecf20Sopenharmony_ci	WREG32(AUX_SW_CONTROL + aux_offset[instance],
1148c2ecf20Sopenharmony_ci	       AUX_SW_WR_BYTES(bytes));
1158c2ecf20Sopenharmony_ci	WREG32(AUX_SW_CONTROL + aux_offset[instance],
1168c2ecf20Sopenharmony_ci	       AUX_SW_WR_BYTES(bytes));
1178c2ecf20Sopenharmony_ci
1188c2ecf20Sopenharmony_ci	/* write the data header into the registers */
1198c2ecf20Sopenharmony_ci	/* request, address, msg size */
1208c2ecf20Sopenharmony_ci	byte = (msg->request << 4) | ((msg->address >> 16) & 0xf);
1218c2ecf20Sopenharmony_ci	WREG32(AUX_SW_DATA + aux_offset[instance],
1228c2ecf20Sopenharmony_ci	       AUX_SW_DATA_MASK(byte) | AUX_SW_AUTOINCREMENT_DISABLE);
1238c2ecf20Sopenharmony_ci
1248c2ecf20Sopenharmony_ci	byte = (msg->address >> 8) & 0xff;
1258c2ecf20Sopenharmony_ci	WREG32(AUX_SW_DATA + aux_offset[instance],
1268c2ecf20Sopenharmony_ci	       AUX_SW_DATA_MASK(byte));
1278c2ecf20Sopenharmony_ci
1288c2ecf20Sopenharmony_ci	byte = msg->address & 0xff;
1298c2ecf20Sopenharmony_ci	WREG32(AUX_SW_DATA + aux_offset[instance],
1308c2ecf20Sopenharmony_ci	       AUX_SW_DATA_MASK(byte));
1318c2ecf20Sopenharmony_ci
1328c2ecf20Sopenharmony_ci	byte = msize;
1338c2ecf20Sopenharmony_ci	WREG32(AUX_SW_DATA + aux_offset[instance],
1348c2ecf20Sopenharmony_ci	       AUX_SW_DATA_MASK(byte));
1358c2ecf20Sopenharmony_ci
1368c2ecf20Sopenharmony_ci	/* if we are writing - write the msg buffer */
1378c2ecf20Sopenharmony_ci	if (is_write) {
1388c2ecf20Sopenharmony_ci		for (i = 0; i < msg->size; i++) {
1398c2ecf20Sopenharmony_ci			WREG32(AUX_SW_DATA + aux_offset[instance],
1408c2ecf20Sopenharmony_ci			       AUX_SW_DATA_MASK(buf[i]));
1418c2ecf20Sopenharmony_ci		}
1428c2ecf20Sopenharmony_ci	}
1438c2ecf20Sopenharmony_ci
1448c2ecf20Sopenharmony_ci	/* clear the ACK */
1458c2ecf20Sopenharmony_ci	WREG32(AUX_SW_INTERRUPT_CONTROL + aux_offset[instance], AUX_SW_DONE_ACK);
1468c2ecf20Sopenharmony_ci
1478c2ecf20Sopenharmony_ci	/* write the size and GO bits */
1488c2ecf20Sopenharmony_ci	WREG32(AUX_SW_CONTROL + aux_offset[instance],
1498c2ecf20Sopenharmony_ci	       AUX_SW_WR_BYTES(bytes) | AUX_SW_GO);
1508c2ecf20Sopenharmony_ci
1518c2ecf20Sopenharmony_ci	/* poll the status registers - TODO irq support */
1528c2ecf20Sopenharmony_ci	do {
1538c2ecf20Sopenharmony_ci		tmp = RREG32(AUX_SW_STATUS + aux_offset[instance]);
1548c2ecf20Sopenharmony_ci		if (tmp & AUX_SW_DONE) {
1558c2ecf20Sopenharmony_ci			break;
1568c2ecf20Sopenharmony_ci		}
1578c2ecf20Sopenharmony_ci		usleep_range(100, 200);
1588c2ecf20Sopenharmony_ci	} while (retry_count++ < 1000);
1598c2ecf20Sopenharmony_ci
1608c2ecf20Sopenharmony_ci	if (retry_count >= 1000) {
1618c2ecf20Sopenharmony_ci		DRM_ERROR("auxch hw never signalled completion, error %08x\n", tmp);
1628c2ecf20Sopenharmony_ci		ret = -EIO;
1638c2ecf20Sopenharmony_ci		goto done;
1648c2ecf20Sopenharmony_ci	}
1658c2ecf20Sopenharmony_ci
1668c2ecf20Sopenharmony_ci	if (tmp & AUX_SW_RX_TIMEOUT) {
1678c2ecf20Sopenharmony_ci		ret = -ETIMEDOUT;
1688c2ecf20Sopenharmony_ci		goto done;
1698c2ecf20Sopenharmony_ci	}
1708c2ecf20Sopenharmony_ci	if (tmp & AUX_RX_ERROR_FLAGS) {
1718c2ecf20Sopenharmony_ci		DRM_DEBUG_KMS_RATELIMITED("dp_aux_ch flags not zero: %08x\n",
1728c2ecf20Sopenharmony_ci					  tmp);
1738c2ecf20Sopenharmony_ci		ret = -EIO;
1748c2ecf20Sopenharmony_ci		goto done;
1758c2ecf20Sopenharmony_ci	}
1768c2ecf20Sopenharmony_ci
1778c2ecf20Sopenharmony_ci	bytes = AUX_SW_REPLY_GET_BYTE_COUNT(tmp);
1788c2ecf20Sopenharmony_ci	if (bytes) {
1798c2ecf20Sopenharmony_ci		WREG32(AUX_SW_DATA + aux_offset[instance],
1808c2ecf20Sopenharmony_ci		       AUX_SW_DATA_RW | AUX_SW_AUTOINCREMENT_DISABLE);
1818c2ecf20Sopenharmony_ci
1828c2ecf20Sopenharmony_ci		tmp = RREG32(AUX_SW_DATA + aux_offset[instance]);
1838c2ecf20Sopenharmony_ci		ack = (tmp >> 8) & 0xff;
1848c2ecf20Sopenharmony_ci
1858c2ecf20Sopenharmony_ci		for (i = 0; i < bytes - 1; i++) {
1868c2ecf20Sopenharmony_ci			tmp = RREG32(AUX_SW_DATA + aux_offset[instance]);
1878c2ecf20Sopenharmony_ci			if (buf)
1888c2ecf20Sopenharmony_ci				buf[i] = (tmp >> 8) & 0xff;
1898c2ecf20Sopenharmony_ci		}
1908c2ecf20Sopenharmony_ci		if (buf)
1918c2ecf20Sopenharmony_ci			ret = bytes - 1;
1928c2ecf20Sopenharmony_ci	}
1938c2ecf20Sopenharmony_ci
1948c2ecf20Sopenharmony_ci	WREG32(AUX_SW_INTERRUPT_CONTROL + aux_offset[instance], AUX_SW_DONE_ACK);
1958c2ecf20Sopenharmony_ci
1968c2ecf20Sopenharmony_ci	if (is_write)
1978c2ecf20Sopenharmony_ci		ret = msg->size;
1988c2ecf20Sopenharmony_cidone:
1998c2ecf20Sopenharmony_ci	mutex_unlock(&chan->mutex);
2008c2ecf20Sopenharmony_ci
2018c2ecf20Sopenharmony_ci	if (ret >= 0)
2028c2ecf20Sopenharmony_ci		msg->reply = ack >> 4;
2038c2ecf20Sopenharmony_ci	return ret;
2048c2ecf20Sopenharmony_ci}
205