Commit 2f60dd50 authored by Luben Tuikov's avatar Luben Tuikov Committed by Alex Deucher

drm/amd: Expose the FRU SMU I2C bus

Expose both SMU I2C buses. Some boards use the same bus for both the RAS
and FRU EEPROMs and others use different buses.  This enables the
additional I2C bus and sets the right buses to use for RAS and FRU EEPROM
access.

Cc: Roy Sun <Roy.Sun@amd.com>
Co-developed-by: default avatarAlex Deucher <Alexander.Deucher@amd.com>
Signed-off-by: default avatarLuben Tuikov <luben.tuikov@amd.com>
Reviewed-by: default avatarAlex Deucher <Alexander.Deucher@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent f06d9e4e
...@@ -75,7 +75,7 @@ static int amdgpu_fru_read_eeprom(struct amdgpu_device *adev, uint32_t addrptr, ...@@ -75,7 +75,7 @@ static int amdgpu_fru_read_eeprom(struct amdgpu_device *adev, uint32_t addrptr,
{ {
int ret, size; int ret, size;
ret = amdgpu_eeprom_read(&adev->pm.smu_i2c, addrptr, buff, 1); ret = amdgpu_eeprom_read(adev->pm.fru_eeprom_i2c_bus, addrptr, buff, 1);
if (ret < 1) { if (ret < 1) {
DRM_WARN("FRU: Failed to get size field"); DRM_WARN("FRU: Failed to get size field");
return ret; return ret;
...@@ -86,7 +86,7 @@ static int amdgpu_fru_read_eeprom(struct amdgpu_device *adev, uint32_t addrptr, ...@@ -86,7 +86,7 @@ static int amdgpu_fru_read_eeprom(struct amdgpu_device *adev, uint32_t addrptr,
*/ */
size = buff[0] - I2C_PRODUCT_INFO_OFFSET; size = buff[0] - I2C_PRODUCT_INFO_OFFSET;
ret = amdgpu_eeprom_read(&adev->pm.smu_i2c, addrptr + 1, buff, size); ret = amdgpu_eeprom_read(adev->pm.fru_eeprom_i2c_bus, addrptr + 1, buff, size);
if (ret < 1) { if (ret < 1) {
DRM_WARN("FRU: Failed to get data field"); DRM_WARN("FRU: Failed to get data field");
return ret; return ret;
...@@ -109,7 +109,7 @@ int amdgpu_fru_get_product_info(struct amdgpu_device *adev) ...@@ -109,7 +109,7 @@ int amdgpu_fru_get_product_info(struct amdgpu_device *adev)
offset = 0; offset = 0;
/* If algo exists, it means that the i2c_adapter's initialized */ /* If algo exists, it means that the i2c_adapter's initialized */
if (!adev->pm.smu_i2c.algo) { if (!adev->pm.fru_eeprom_i2c_bus || !adev->pm.fru_eeprom_i2c_bus->algo) {
DRM_WARN("Cannot access FRU, EEPROM accessor not initialized"); DRM_WARN("Cannot access FRU, EEPROM accessor not initialized");
return -ENODEV; return -ENODEV;
} }
......
...@@ -194,7 +194,7 @@ static int __write_table_header(struct amdgpu_ras_eeprom_control *control) ...@@ -194,7 +194,7 @@ static int __write_table_header(struct amdgpu_ras_eeprom_control *control)
/* i2c may be unstable in gpu reset */ /* i2c may be unstable in gpu reset */
down_read(&adev->reset_sem); down_read(&adev->reset_sem);
res = amdgpu_eeprom_write(&adev->pm.smu_i2c, res = amdgpu_eeprom_write(adev->pm.ras_eeprom_i2c_bus,
control->i2c_address + control->i2c_address +
control->ras_header_offset, control->ras_header_offset,
buf, RAS_TABLE_HEADER_SIZE); buf, RAS_TABLE_HEADER_SIZE);
...@@ -389,7 +389,7 @@ static int __amdgpu_ras_eeprom_write(struct amdgpu_ras_eeprom_control *control, ...@@ -389,7 +389,7 @@ static int __amdgpu_ras_eeprom_write(struct amdgpu_ras_eeprom_control *control,
/* i2c may be unstable in gpu reset */ /* i2c may be unstable in gpu reset */
down_read(&adev->reset_sem); down_read(&adev->reset_sem);
buf_size = num * RAS_TABLE_RECORD_SIZE; buf_size = num * RAS_TABLE_RECORD_SIZE;
res = amdgpu_eeprom_write(&adev->pm.smu_i2c, res = amdgpu_eeprom_write(adev->pm.ras_eeprom_i2c_bus,
control->i2c_address + control->i2c_address +
RAS_INDEX_TO_OFFSET(control, fri), RAS_INDEX_TO_OFFSET(control, fri),
buf, buf_size); buf, buf_size);
...@@ -548,7 +548,7 @@ amdgpu_ras_eeprom_update_header(struct amdgpu_ras_eeprom_control *control) ...@@ -548,7 +548,7 @@ amdgpu_ras_eeprom_update_header(struct amdgpu_ras_eeprom_control *control)
} }
down_read(&adev->reset_sem); down_read(&adev->reset_sem);
res = amdgpu_eeprom_read(&adev->pm.smu_i2c, res = amdgpu_eeprom_read(adev->pm.ras_eeprom_i2c_bus,
control->i2c_address + control->i2c_address +
control->ras_record_offset, control->ras_record_offset,
buf, buf_size); buf, buf_size);
...@@ -644,7 +644,7 @@ static int __amdgpu_ras_eeprom_read(struct amdgpu_ras_eeprom_control *control, ...@@ -644,7 +644,7 @@ static int __amdgpu_ras_eeprom_read(struct amdgpu_ras_eeprom_control *control,
/* i2c may be unstable in gpu reset */ /* i2c may be unstable in gpu reset */
down_read(&adev->reset_sem); down_read(&adev->reset_sem);
buf_size = num * RAS_TABLE_RECORD_SIZE; buf_size = num * RAS_TABLE_RECORD_SIZE;
res = amdgpu_eeprom_read(&adev->pm.smu_i2c, res = amdgpu_eeprom_read(adev->pm.ras_eeprom_i2c_bus,
control->i2c_address + control->i2c_address +
RAS_INDEX_TO_OFFSET(control, fri), RAS_INDEX_TO_OFFSET(control, fri),
buf, buf_size); buf, buf_size);
...@@ -1009,7 +1009,7 @@ static int __verify_ras_table_checksum(struct amdgpu_ras_eeprom_control *control ...@@ -1009,7 +1009,7 @@ static int __verify_ras_table_checksum(struct amdgpu_ras_eeprom_control *control
return -ENOMEM; return -ENOMEM;
} }
res = amdgpu_eeprom_read(&adev->pm.smu_i2c, res = amdgpu_eeprom_read(adev->pm.ras_eeprom_i2c_bus,
control->i2c_address + control->i2c_address +
control->ras_header_offset, control->ras_header_offset,
buf, buf_size); buf, buf_size);
...@@ -1045,7 +1045,7 @@ int amdgpu_ras_eeprom_init(struct amdgpu_ras_eeprom_control *control, ...@@ -1045,7 +1045,7 @@ int amdgpu_ras_eeprom_init(struct amdgpu_ras_eeprom_control *control,
return 0; return 0;
/* Verify i2c adapter is initialized */ /* Verify i2c adapter is initialized */
if (!adev->pm.smu_i2c.algo) if (!adev->pm.ras_eeprom_i2c_bus || !adev->pm.ras_eeprom_i2c_bus->algo)
return -ENOENT; return -ENOENT;
if (!__get_eeprom_i2c_addr(adev, control)) if (!__get_eeprom_i2c_addr(adev, control))
...@@ -1057,7 +1057,7 @@ int amdgpu_ras_eeprom_init(struct amdgpu_ras_eeprom_control *control, ...@@ -1057,7 +1057,7 @@ int amdgpu_ras_eeprom_init(struct amdgpu_ras_eeprom_control *control,
mutex_init(&control->ras_tbl_mutex); mutex_init(&control->ras_tbl_mutex);
/* Read the table header from EEPROM address */ /* Read the table header from EEPROM address */
res = amdgpu_eeprom_read(&adev->pm.smu_i2c, res = amdgpu_eeprom_read(adev->pm.ras_eeprom_i2c_bus,
control->i2c_address + control->ras_header_offset, control->i2c_address + control->ras_header_offset,
buf, RAS_TABLE_HEADER_SIZE); buf, RAS_TABLE_HEADER_SIZE);
if (res < RAS_TABLE_HEADER_SIZE) { if (res < RAS_TABLE_HEADER_SIZE) {
......
...@@ -26,6 +26,7 @@ ...@@ -26,6 +26,7 @@
#include "smu_v11_0_i2c.h" #include "smu_v11_0_i2c.h"
#include "amdgpu.h" #include "amdgpu.h"
#include "amdgpu_dpm.h"
#include "soc15_common.h" #include "soc15_common.h"
#include <drm/drm_fixed.h> #include <drm/drm_fixed.h>
#include <drm/drm_drv.h> #include <drm/drm_drv.h>
...@@ -43,11 +44,10 @@ ...@@ -43,11 +44,10 @@
#define I2C_X_RESTART BIT(31) #define I2C_X_RESTART BIT(31)
#define to_amdgpu_device(x) (container_of(x, struct amdgpu_device, pm.smu_i2c))
static void smu_v11_0_i2c_set_clock_gating(struct i2c_adapter *control, bool en) static void smu_v11_0_i2c_set_clock_gating(struct i2c_adapter *control, bool en)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
uint32_t reg = RREG32_SOC15(SMUIO, 0, mmSMUIO_PWRMGT); uint32_t reg = RREG32_SOC15(SMUIO, 0, mmSMUIO_PWRMGT);
reg = REG_SET_FIELD(reg, SMUIO_PWRMGT, i2c_clk_gate_en, en ? 1 : 0); reg = REG_SET_FIELD(reg, SMUIO_PWRMGT, i2c_clk_gate_en, en ? 1 : 0);
...@@ -75,7 +75,8 @@ static void smu_v11_0_i2c_set_clock_gating(struct i2c_adapter *control, bool en) ...@@ -75,7 +75,8 @@ static void smu_v11_0_i2c_set_clock_gating(struct i2c_adapter *control, bool en)
static int smu_v11_0_i2c_enable(struct i2c_adapter *control, bool enable) static int smu_v11_0_i2c_enable(struct i2c_adapter *control, bool enable)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
WREG32_SOC15(SMUIO, 0, mmCKSVII2C_IC_ENABLE, enable ? 1 : 0); WREG32_SOC15(SMUIO, 0, mmCKSVII2C_IC_ENABLE, enable ? 1 : 0);
...@@ -100,7 +101,8 @@ static int smu_v11_0_i2c_enable(struct i2c_adapter *control, bool enable) ...@@ -100,7 +101,8 @@ static int smu_v11_0_i2c_enable(struct i2c_adapter *control, bool enable)
static void smu_v11_0_i2c_clear_status(struct i2c_adapter *control) static void smu_v11_0_i2c_clear_status(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
/* do */ /* do */
{ {
RREG32_SOC15(SMUIO, 0, mmCKSVII2C_IC_CLR_INTR); RREG32_SOC15(SMUIO, 0, mmCKSVII2C_IC_CLR_INTR);
...@@ -110,7 +112,8 @@ static void smu_v11_0_i2c_clear_status(struct i2c_adapter *control) ...@@ -110,7 +112,8 @@ static void smu_v11_0_i2c_clear_status(struct i2c_adapter *control)
static void smu_v11_0_i2c_configure(struct i2c_adapter *control) static void smu_v11_0_i2c_configure(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
uint32_t reg = 0; uint32_t reg = 0;
reg = REG_SET_FIELD(reg, CKSVII2C_IC_CON, IC_SLAVE_DISABLE, 1); reg = REG_SET_FIELD(reg, CKSVII2C_IC_CON, IC_SLAVE_DISABLE, 1);
...@@ -131,7 +134,8 @@ static void smu_v11_0_i2c_configure(struct i2c_adapter *control) ...@@ -131,7 +134,8 @@ static void smu_v11_0_i2c_configure(struct i2c_adapter *control)
static void smu_v11_0_i2c_set_clock(struct i2c_adapter *control) static void smu_v11_0_i2c_set_clock(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
/* /*
* Standard mode speed, These values are taken from SMUIO MAS, * Standard mode speed, These values are taken from SMUIO MAS,
...@@ -154,7 +158,8 @@ static void smu_v11_0_i2c_set_clock(struct i2c_adapter *control) ...@@ -154,7 +158,8 @@ static void smu_v11_0_i2c_set_clock(struct i2c_adapter *control)
static void smu_v11_0_i2c_set_address(struct i2c_adapter *control, u16 address) static void smu_v11_0_i2c_set_address(struct i2c_adapter *control, u16 address)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
/* The IC_TAR::IC_TAR field is 10-bits wide. /* The IC_TAR::IC_TAR field is 10-bits wide.
* It takes a 7-bit or 10-bit addresses as an address, * It takes a 7-bit or 10-bit addresses as an address,
...@@ -165,7 +170,8 @@ static void smu_v11_0_i2c_set_address(struct i2c_adapter *control, u16 address) ...@@ -165,7 +170,8 @@ static void smu_v11_0_i2c_set_address(struct i2c_adapter *control, u16 address)
static uint32_t smu_v11_0_i2c_poll_tx_status(struct i2c_adapter *control) static uint32_t smu_v11_0_i2c_poll_tx_status(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
uint32_t ret = I2C_OK; uint32_t ret = I2C_OK;
uint32_t reg, reg_c_tx_abrt_source; uint32_t reg, reg_c_tx_abrt_source;
...@@ -216,7 +222,8 @@ static uint32_t smu_v11_0_i2c_poll_tx_status(struct i2c_adapter *control) ...@@ -216,7 +222,8 @@ static uint32_t smu_v11_0_i2c_poll_tx_status(struct i2c_adapter *control)
static uint32_t smu_v11_0_i2c_poll_rx_status(struct i2c_adapter *control) static uint32_t smu_v11_0_i2c_poll_rx_status(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
uint32_t ret = I2C_OK; uint32_t ret = I2C_OK;
uint32_t reg_ic_status, reg_c_tx_abrt_source; uint32_t reg_ic_status, reg_c_tx_abrt_source;
...@@ -262,7 +269,8 @@ static uint32_t smu_v11_0_i2c_transmit(struct i2c_adapter *control, ...@@ -262,7 +269,8 @@ static uint32_t smu_v11_0_i2c_transmit(struct i2c_adapter *control,
u16 address, u8 *data, u16 address, u8 *data,
u32 numbytes, u32 i2c_flag) u32 numbytes, u32 i2c_flag)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
u32 bytes_sent, reg, ret = I2C_OK; u32 bytes_sent, reg, ret = I2C_OK;
unsigned long timeout_counter; unsigned long timeout_counter;
...@@ -360,7 +368,8 @@ static uint32_t smu_v11_0_i2c_receive(struct i2c_adapter *control, ...@@ -360,7 +368,8 @@ static uint32_t smu_v11_0_i2c_receive(struct i2c_adapter *control,
u16 address, u8 *data, u16 address, u8 *data,
u32 numbytes, u32 i2c_flag) u32 numbytes, u32 i2c_flag)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
uint32_t bytes_received, ret = I2C_OK; uint32_t bytes_received, ret = I2C_OK;
bytes_received = 0; bytes_received = 0;
...@@ -431,7 +440,8 @@ static uint32_t smu_v11_0_i2c_receive(struct i2c_adapter *control, ...@@ -431,7 +440,8 @@ static uint32_t smu_v11_0_i2c_receive(struct i2c_adapter *control,
static void smu_v11_0_i2c_abort(struct i2c_adapter *control) static void smu_v11_0_i2c_abort(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
uint32_t reg = 0; uint32_t reg = 0;
/* Enable I2C engine; */ /* Enable I2C engine; */
...@@ -447,7 +457,8 @@ static void smu_v11_0_i2c_abort(struct i2c_adapter *control) ...@@ -447,7 +457,8 @@ static void smu_v11_0_i2c_abort(struct i2c_adapter *control)
static bool smu_v11_0_i2c_activity_done(struct i2c_adapter *control) static bool smu_v11_0_i2c_activity_done(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
const uint32_t IDLE_TIMEOUT = 1024; const uint32_t IDLE_TIMEOUT = 1024;
uint32_t timeout_count = 0; uint32_t timeout_count = 0;
...@@ -508,7 +519,8 @@ static void smu_v11_0_i2c_init(struct i2c_adapter *control) ...@@ -508,7 +519,8 @@ static void smu_v11_0_i2c_init(struct i2c_adapter *control)
static void smu_v11_0_i2c_fini(struct i2c_adapter *control) static void smu_v11_0_i2c_fini(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
u32 status, enable, en_stat; u32 status, enable, en_stat;
int res; int res;
...@@ -543,7 +555,8 @@ static void smu_v11_0_i2c_fini(struct i2c_adapter *control) ...@@ -543,7 +555,8 @@ static void smu_v11_0_i2c_fini(struct i2c_adapter *control)
static bool smu_v11_0_i2c_bus_lock(struct i2c_adapter *control) static bool smu_v11_0_i2c_bus_lock(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
/* Send PPSMC_MSG_RequestI2CBus */ /* Send PPSMC_MSG_RequestI2CBus */
if (!amdgpu_dpm_smu_i2c_bus_access(adev, true)) if (!amdgpu_dpm_smu_i2c_bus_access(adev, true))
...@@ -554,7 +567,8 @@ static bool smu_v11_0_i2c_bus_lock(struct i2c_adapter *control) ...@@ -554,7 +567,8 @@ static bool smu_v11_0_i2c_bus_lock(struct i2c_adapter *control)
static bool smu_v11_0_i2c_bus_unlock(struct i2c_adapter *control) static bool smu_v11_0_i2c_bus_unlock(struct i2c_adapter *control)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(control);
struct amdgpu_device *adev = smu_i2c->adev;
/* Send PPSMC_MSG_ReleaseI2CBus */ /* Send PPSMC_MSG_ReleaseI2CBus */
if (!amdgpu_dpm_smu_i2c_bus_access(adev, false)) if (!amdgpu_dpm_smu_i2c_bus_access(adev, false))
...@@ -594,9 +608,10 @@ static uint32_t smu_v11_0_i2c_write_data(struct i2c_adapter *control, ...@@ -594,9 +608,10 @@ static uint32_t smu_v11_0_i2c_write_data(struct i2c_adapter *control,
static void lock_bus(struct i2c_adapter *i2c, unsigned int flags) static void lock_bus(struct i2c_adapter *i2c, unsigned int flags)
{ {
struct amdgpu_device *adev = to_amdgpu_device(i2c); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(i2c);
struct amdgpu_device *adev = smu_i2c->adev;
mutex_lock(&adev->pm.smu_i2c_mutex); mutex_lock(&smu_i2c->mutex);
if (!smu_v11_0_i2c_bus_lock(i2c)) if (!smu_v11_0_i2c_bus_lock(i2c))
DRM_ERROR("Failed to lock the bus from SMU"); DRM_ERROR("Failed to lock the bus from SMU");
else else
...@@ -611,13 +626,14 @@ static int trylock_bus(struct i2c_adapter *i2c, unsigned int flags) ...@@ -611,13 +626,14 @@ static int trylock_bus(struct i2c_adapter *i2c, unsigned int flags)
static void unlock_bus(struct i2c_adapter *i2c, unsigned int flags) static void unlock_bus(struct i2c_adapter *i2c, unsigned int flags)
{ {
struct amdgpu_device *adev = to_amdgpu_device(i2c); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(i2c);
struct amdgpu_device *adev = smu_i2c->adev;
if (!smu_v11_0_i2c_bus_unlock(i2c)) if (!smu_v11_0_i2c_bus_unlock(i2c))
DRM_ERROR("Failed to unlock the bus from SMU"); DRM_ERROR("Failed to unlock the bus from SMU");
else else
adev->pm.bus_locked = false; adev->pm.bus_locked = false;
mutex_unlock(&adev->pm.smu_i2c_mutex); mutex_unlock(&smu_i2c->mutex);
} }
static const struct i2c_lock_operations smu_v11_0_i2c_i2c_lock_ops = { static const struct i2c_lock_operations smu_v11_0_i2c_i2c_lock_ops = {
...@@ -706,19 +722,26 @@ static const struct i2c_adapter_quirks smu_v11_0_i2c_control_quirks = { ...@@ -706,19 +722,26 @@ static const struct i2c_adapter_quirks smu_v11_0_i2c_control_quirks = {
.flags = I2C_AQ_NO_ZERO_LEN, .flags = I2C_AQ_NO_ZERO_LEN,
}; };
int smu_v11_0_i2c_control_init(struct i2c_adapter *control) int smu_v11_0_i2c_control_init(struct amdgpu_device *adev)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[0];
struct i2c_adapter *control = &smu_i2c->adapter;
int res; int res;
mutex_init(&adev->pm.smu_i2c_mutex); smu_i2c->adev = adev;
smu_i2c->port = 0;
mutex_init(&smu_i2c->mutex);
control->owner = THIS_MODULE; control->owner = THIS_MODULE;
control->class = I2C_CLASS_HWMON; control->class = I2C_CLASS_HWMON;
control->dev.parent = &adev->pdev->dev; control->dev.parent = &adev->pdev->dev;
control->algo = &smu_v11_0_i2c_algo; control->algo = &smu_v11_0_i2c_algo;
snprintf(control->name, sizeof(control->name), "AMDGPU SMU"); snprintf(control->name, sizeof(control->name), "AMDGPU SMU 0");
control->lock_ops = &smu_v11_0_i2c_i2c_lock_ops; control->lock_ops = &smu_v11_0_i2c_i2c_lock_ops;
control->quirks = &smu_v11_0_i2c_control_quirks; control->quirks = &smu_v11_0_i2c_control_quirks;
i2c_set_adapdata(control, smu_i2c);
adev->pm.ras_eeprom_i2c_bus = &adev->pm.smu_i2c[0].adapter;
adev->pm.fru_eeprom_i2c_bus = NULL;
res = i2c_add_adapter(control); res = i2c_add_adapter(control);
if (res) if (res)
...@@ -727,9 +750,12 @@ int smu_v11_0_i2c_control_init(struct i2c_adapter *control) ...@@ -727,9 +750,12 @@ int smu_v11_0_i2c_control_init(struct i2c_adapter *control)
return res; return res;
} }
void smu_v11_0_i2c_control_fini(struct i2c_adapter *control) void smu_v11_0_i2c_control_fini(struct amdgpu_device *adev)
{ {
struct i2c_adapter *control = adev->pm.ras_eeprom_i2c_bus;
i2c_del_adapter(control); i2c_del_adapter(control);
adev->pm.ras_eeprom_i2c_bus = NULL;
} }
/* /*
......
...@@ -26,9 +26,9 @@ ...@@ -26,9 +26,9 @@
#include <linux/types.h> #include <linux/types.h>
struct i2c_adapter; struct amdgpu_device;
int smu_v11_0_i2c_control_init(struct i2c_adapter *control); int smu_v11_0_i2c_control_init(struct amdgpu_device *adev);
void smu_v11_0_i2c_control_fini(struct i2c_adapter *control); void smu_v11_0_i2c_control_fini(struct amdgpu_device *adev);
#endif #endif
...@@ -284,6 +284,15 @@ enum ip_power_state { ...@@ -284,6 +284,15 @@ enum ip_power_state {
/* Used to mask smu debug modes */ /* Used to mask smu debug modes */
#define SMU_DEBUG_HALT_ON_ERROR 0x1 #define SMU_DEBUG_HALT_ON_ERROR 0x1
#define MAX_SMU_I2C_BUSES 2
struct amdgpu_smu_i2c_bus {
struct i2c_adapter adapter;
struct amdgpu_device *adev;
int port;
struct mutex mutex;
};
struct amdgpu_pm { struct amdgpu_pm {
struct mutex mutex; struct mutex mutex;
u32 current_sclk; u32 current_sclk;
...@@ -316,8 +325,9 @@ struct amdgpu_pm { ...@@ -316,8 +325,9 @@ struct amdgpu_pm {
uint32_t pp_feature; uint32_t pp_feature;
/* Used for I2C access to various EEPROMs on relevant ASICs */ /* Used for I2C access to various EEPROMs on relevant ASICs */
struct i2c_adapter smu_i2c; struct amdgpu_smu_i2c_bus smu_i2c[MAX_SMU_I2C_BUSES];
struct mutex smu_i2c_mutex; struct i2c_adapter *ras_eeprom_i2c_bus;
struct i2c_adapter *fru_eeprom_i2c_bus;
struct list_head pm_attr_list; struct list_head pm_attr_list;
atomic_t pwr_state[AMD_IP_BLOCK_TYPE_NUM]; atomic_t pwr_state[AMD_IP_BLOCK_TYPE_NUM];
......
...@@ -520,7 +520,7 @@ static int vega20_smu_init(struct pp_hwmgr *hwmgr) ...@@ -520,7 +520,7 @@ static int vega20_smu_init(struct pp_hwmgr *hwmgr)
priv->smu_tables.entry[TABLE_ACTIVITY_MONITOR_COEFF].version = 0x01; priv->smu_tables.entry[TABLE_ACTIVITY_MONITOR_COEFF].version = 0x01;
priv->smu_tables.entry[TABLE_ACTIVITY_MONITOR_COEFF].size = sizeof(DpmActivityMonitorCoeffInt_t); priv->smu_tables.entry[TABLE_ACTIVITY_MONITOR_COEFF].size = sizeof(DpmActivityMonitorCoeffInt_t);
ret = smu_v11_0_i2c_control_init(&adev->pm.smu_i2c); ret = smu_v11_0_i2c_control_init(adev);
if (ret) if (ret)
goto err4; goto err4;
...@@ -558,7 +558,7 @@ static int vega20_smu_fini(struct pp_hwmgr *hwmgr) ...@@ -558,7 +558,7 @@ static int vega20_smu_fini(struct pp_hwmgr *hwmgr)
(struct vega20_smumgr *)(hwmgr->smu_backend); (struct vega20_smumgr *)(hwmgr->smu_backend);
struct amdgpu_device *adev = hwmgr->adev; struct amdgpu_device *adev = hwmgr->adev;
smu_v11_0_i2c_control_fini(&adev->pm.smu_i2c); smu_v11_0_i2c_control_fini(adev);
if (priv) { if (priv) {
amdgpu_bo_free_kernel(&priv->smu_tables.entry[TABLE_PPTABLE].handle, amdgpu_bo_free_kernel(&priv->smu_tables.entry[TABLE_PPTABLE].handle,
......
...@@ -886,7 +886,7 @@ static int smu_smc_table_sw_init(struct smu_context *smu) ...@@ -886,7 +886,7 @@ static int smu_smc_table_sw_init(struct smu_context *smu)
if (ret) if (ret)
return ret; return ret;
ret = smu_i2c_init(smu, &smu->adev->pm.smu_i2c); ret = smu_i2c_init(smu);
if (ret) if (ret)
return ret; return ret;
...@@ -897,7 +897,7 @@ static int smu_smc_table_sw_fini(struct smu_context *smu) ...@@ -897,7 +897,7 @@ static int smu_smc_table_sw_fini(struct smu_context *smu)
{ {
int ret; int ret;
smu_i2c_fini(smu, &smu->adev->pm.smu_i2c); smu_i2c_fini(smu);
smu_free_dummy_read_table(smu); smu_free_dummy_read_table(smu);
......
...@@ -816,12 +816,12 @@ struct pptable_funcs { ...@@ -816,12 +816,12 @@ struct pptable_funcs {
* other devices. The i2c's EEPROM also stores bad page tables on boards * other devices. The i2c's EEPROM also stores bad page tables on boards
* with ECC. * with ECC.
*/ */
int (*i2c_init)(struct smu_context *smu, struct i2c_adapter *control); int (*i2c_init)(struct smu_context *smu);
/** /**
* @i2c_fini: Tear down i2c. * @i2c_fini: Tear down i2c.
*/ */
void (*i2c_fini)(struct smu_context *smu, struct i2c_adapter *control); void (*i2c_fini)(struct smu_context *smu);
/** /**
* @get_unique_id: Get the GPU's unique id. Used for asset tracking. * @get_unique_id: Get the GPU's unique id. Used for asset tracking.
......
...@@ -25,6 +25,7 @@ ...@@ -25,6 +25,7 @@
#include <linux/firmware.h> #include <linux/firmware.h>
#include "amdgpu.h" #include "amdgpu.h"
#include "amdgpu_dpm.h"
#include "amdgpu_smu.h" #include "amdgpu_smu.h"
#include "atomfirmware.h" #include "atomfirmware.h"
#include "amdgpu_atomfirmware.h" #include "amdgpu_atomfirmware.h"
...@@ -45,6 +46,7 @@ ...@@ -45,6 +46,7 @@
#include <linux/pci.h> #include <linux/pci.h>
#include "amdgpu_ras.h" #include "amdgpu_ras.h"
#include "smu_cmn.h" #include "smu_cmn.h"
#include "amdgpu_dpm.h"
/* /*
* DO NOT use these for err/warn/info/debug messages. * DO NOT use these for err/warn/info/debug messages.
...@@ -56,8 +58,6 @@ ...@@ -56,8 +58,6 @@
#undef pr_info #undef pr_info
#undef pr_debug #undef pr_debug
#define to_amdgpu_device(x) (container_of(x, struct amdgpu_device, pm.smu_i2c))
#define ARCTURUS_FEA_MAP(smu_feature, arcturus_feature) \ #define ARCTURUS_FEA_MAP(smu_feature, arcturus_feature) \
[smu_feature] = {1, (arcturus_feature)} [smu_feature] = {1, (arcturus_feature)}
...@@ -2062,7 +2062,8 @@ static int arcturus_dpm_set_vcn_enable(struct smu_context *smu, bool enable) ...@@ -2062,7 +2062,8 @@ static int arcturus_dpm_set_vcn_enable(struct smu_context *smu, bool enable)
static int arcturus_i2c_xfer(struct i2c_adapter *i2c_adap, static int arcturus_i2c_xfer(struct i2c_adapter *i2c_adap,
struct i2c_msg *msg, int num_msgs) struct i2c_msg *msg, int num_msgs)
{ {
struct amdgpu_device *adev = to_amdgpu_device(i2c_adap); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(i2c_adap);
struct amdgpu_device *adev = smu_i2c->adev;
struct smu_context *smu = adev->powerplay.pp_handle; struct smu_context *smu = adev->powerplay.pp_handle;
struct smu_table_context *smu_table = &smu->smu_table; struct smu_table_context *smu_table = &smu->smu_table;
struct smu_table *table = &smu_table->driver_table; struct smu_table *table = &smu_table->driver_table;
...@@ -2074,7 +2075,7 @@ static int arcturus_i2c_xfer(struct i2c_adapter *i2c_adap, ...@@ -2074,7 +2075,7 @@ static int arcturus_i2c_xfer(struct i2c_adapter *i2c_adap,
if (!req) if (!req)
return -ENOMEM; return -ENOMEM;
req->I2CcontrollerPort = 0; req->I2CcontrollerPort = smu_i2c->port;
req->I2CSpeed = I2C_SPEED_FAST_400K; req->I2CSpeed = I2C_SPEED_FAST_400K;
req->SlaveAddress = msg[0].addr << 1; /* wants an 8-bit address */ req->SlaveAddress = msg[0].addr << 1; /* wants an 8-bit address */
dir = msg[0].flags & I2C_M_RD; dir = msg[0].flags & I2C_M_RD;
...@@ -2153,28 +2154,60 @@ static const struct i2c_adapter_quirks arcturus_i2c_control_quirks = { ...@@ -2153,28 +2154,60 @@ static const struct i2c_adapter_quirks arcturus_i2c_control_quirks = {
.max_comb_2nd_msg_len = MAX_SW_I2C_COMMANDS - 2, .max_comb_2nd_msg_len = MAX_SW_I2C_COMMANDS - 2,
}; };
static int arcturus_i2c_control_init(struct smu_context *smu, struct i2c_adapter *control) static int arcturus_i2c_control_init(struct smu_context *smu)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_device *adev = smu->adev;
int res; int res, i;
for (i = 0; i < MAX_SMU_I2C_BUSES; i++) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
smu_i2c->adev = adev;
smu_i2c->port = i;
mutex_init(&smu_i2c->mutex);
control->owner = THIS_MODULE; control->owner = THIS_MODULE;
control->class = I2C_CLASS_HWMON; control->class = I2C_CLASS_HWMON;
control->dev.parent = &adev->pdev->dev; control->dev.parent = &adev->pdev->dev;
control->algo = &arcturus_i2c_algo; control->algo = &arcturus_i2c_algo;
control->quirks = &arcturus_i2c_control_quirks; control->quirks = &arcturus_i2c_control_quirks;
snprintf(control->name, sizeof(control->name), "AMDGPU SMU"); snprintf(control->name, sizeof(control->name), "AMDGPU SMU %d", i);
i2c_set_adapdata(control, smu_i2c);
res = i2c_add_adapter(control); res = i2c_add_adapter(control);
if (res) if (res) {
DRM_ERROR("Failed to register hw i2c, err: %d\n", res); DRM_ERROR("Failed to register hw i2c, err: %d\n", res);
goto Out_err;
}
}
adev->pm.ras_eeprom_i2c_bus = &adev->pm.smu_i2c[0].adapter;
adev->pm.fru_eeprom_i2c_bus = &adev->pm.smu_i2c[1].adapter;
return 0;
Out_err:
for ( ; i >= 0; i--) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
i2c_del_adapter(control);
}
return res; return res;
} }
static void arcturus_i2c_control_fini(struct smu_context *smu, struct i2c_adapter *control) static void arcturus_i2c_control_fini(struct smu_context *smu)
{ {
struct amdgpu_device *adev = smu->adev;
int i;
for (i = 0; i < MAX_SMU_I2C_BUSES; i++) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
i2c_del_adapter(control); i2c_del_adapter(control);
}
adev->pm.ras_eeprom_i2c_bus = NULL;
adev->pm.fru_eeprom_i2c_bus = NULL;
} }
static void arcturus_get_unique_id(struct smu_context *smu) static void arcturus_get_unique_id(struct smu_context *smu)
......
...@@ -27,6 +27,7 @@ ...@@ -27,6 +27,7 @@
#include <linux/pci.h> #include <linux/pci.h>
#include <linux/i2c.h> #include <linux/i2c.h>
#include "amdgpu.h" #include "amdgpu.h"
#include "amdgpu_dpm.h"
#include "amdgpu_smu.h" #include "amdgpu_smu.h"
#include "atomfirmware.h" #include "atomfirmware.h"
#include "amdgpu_atomfirmware.h" #include "amdgpu_atomfirmware.h"
...@@ -57,8 +58,6 @@ ...@@ -57,8 +58,6 @@
#undef pr_info #undef pr_info
#undef pr_debug #undef pr_debug
#define to_amdgpu_device(x) (container_of(x, struct amdgpu_device, pm.smu_i2c))
#define FEATURE_MASK(feature) (1ULL << feature) #define FEATURE_MASK(feature) (1ULL << feature)
#define SMC_DPM_FEATURE ( \ #define SMC_DPM_FEATURE ( \
FEATURE_MASK(FEATURE_DPM_PREFETCHER_BIT) | \ FEATURE_MASK(FEATURE_DPM_PREFETCHER_BIT) | \
...@@ -2746,7 +2745,8 @@ static ssize_t navi10_get_legacy_gpu_metrics(struct smu_context *smu, ...@@ -2746,7 +2745,8 @@ static ssize_t navi10_get_legacy_gpu_metrics(struct smu_context *smu,
static int navi10_i2c_xfer(struct i2c_adapter *i2c_adap, static int navi10_i2c_xfer(struct i2c_adapter *i2c_adap,
struct i2c_msg *msg, int num_msgs) struct i2c_msg *msg, int num_msgs)
{ {
struct amdgpu_device *adev = to_amdgpu_device(i2c_adap); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(i2c_adap);
struct amdgpu_device *adev = smu_i2c->adev;
struct smu_context *smu = adev->powerplay.pp_handle; struct smu_context *smu = adev->powerplay.pp_handle;
struct smu_table_context *smu_table = &smu->smu_table; struct smu_table_context *smu_table = &smu->smu_table;
struct smu_table *table = &smu_table->driver_table; struct smu_table *table = &smu_table->driver_table;
...@@ -2758,7 +2758,7 @@ static int navi10_i2c_xfer(struct i2c_adapter *i2c_adap, ...@@ -2758,7 +2758,7 @@ static int navi10_i2c_xfer(struct i2c_adapter *i2c_adap,
if (!req) if (!req)
return -ENOMEM; return -ENOMEM;
req->I2CcontrollerPort = 0; req->I2CcontrollerPort = smu_i2c->port;
req->I2CSpeed = I2C_SPEED_FAST_400K; req->I2CSpeed = I2C_SPEED_FAST_400K;
req->SlaveAddress = msg[0].addr << 1; /* wants an 8-bit address */ req->SlaveAddress = msg[0].addr << 1; /* wants an 8-bit address */
dir = msg[0].flags & I2C_M_RD; dir = msg[0].flags & I2C_M_RD;
...@@ -2836,28 +2836,60 @@ static const struct i2c_adapter_quirks navi10_i2c_control_quirks = { ...@@ -2836,28 +2836,60 @@ static const struct i2c_adapter_quirks navi10_i2c_control_quirks = {
.max_comb_2nd_msg_len = MAX_SW_I2C_COMMANDS - 2, .max_comb_2nd_msg_len = MAX_SW_I2C_COMMANDS - 2,
}; };
static int navi10_i2c_control_init(struct smu_context *smu, struct i2c_adapter *control) static int navi10_i2c_control_init(struct smu_context *smu)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_device *adev = smu->adev;
int res; int res, i;
for (i = 0; i < MAX_SMU_I2C_BUSES; i++) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
smu_i2c->adev = adev;
smu_i2c->port = i;
mutex_init(&smu_i2c->mutex);
control->owner = THIS_MODULE; control->owner = THIS_MODULE;
control->class = I2C_CLASS_HWMON; control->class = I2C_CLASS_HWMON;
control->dev.parent = &adev->pdev->dev; control->dev.parent = &adev->pdev->dev;
control->algo = &navi10_i2c_algo; control->algo = &navi10_i2c_algo;
snprintf(control->name, sizeof(control->name), "AMDGPU SMU"); snprintf(control->name, sizeof(control->name), "AMDGPU SMU %d", i);
control->quirks = &navi10_i2c_control_quirks; control->quirks = &navi10_i2c_control_quirks;
i2c_set_adapdata(control, smu_i2c);
res = i2c_add_adapter(control); res = i2c_add_adapter(control);
if (res) if (res) {
DRM_ERROR("Failed to register hw i2c, err: %d\n", res); DRM_ERROR("Failed to register hw i2c, err: %d\n", res);
goto Out_err;
}
}
adev->pm.ras_eeprom_i2c_bus = &adev->pm.smu_i2c[0].adapter;
adev->pm.fru_eeprom_i2c_bus = &adev->pm.smu_i2c[1].adapter;
return 0;
Out_err:
for ( ; i >= 0; i--) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
i2c_del_adapter(control);
}
return res; return res;
} }
static void navi10_i2c_control_fini(struct smu_context *smu, struct i2c_adapter *control) static void navi10_i2c_control_fini(struct smu_context *smu)
{ {
struct amdgpu_device *adev = smu->adev;
int i;
for (i = 0; i < MAX_SMU_I2C_BUSES; i++) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
i2c_del_adapter(control); i2c_del_adapter(control);
}
adev->pm.ras_eeprom_i2c_bus = NULL;
adev->pm.fru_eeprom_i2c_bus = NULL;
} }
static ssize_t navi10_get_gpu_metrics(struct smu_context *smu, static ssize_t navi10_get_gpu_metrics(struct smu_context *smu,
......
...@@ -27,6 +27,7 @@ ...@@ -27,6 +27,7 @@
#include <linux/pci.h> #include <linux/pci.h>
#include <linux/i2c.h> #include <linux/i2c.h>
#include "amdgpu.h" #include "amdgpu.h"
#include "amdgpu_dpm.h"
#include "amdgpu_smu.h" #include "amdgpu_smu.h"
#include "atomfirmware.h" #include "atomfirmware.h"
#include "amdgpu_atomfirmware.h" #include "amdgpu_atomfirmware.h"
...@@ -59,8 +60,6 @@ ...@@ -59,8 +60,6 @@
#undef pr_info #undef pr_info
#undef pr_debug #undef pr_debug
#define to_amdgpu_device(x) (container_of(x, struct amdgpu_device, pm.smu_i2c))
#define FEATURE_MASK(feature) (1ULL << feature) #define FEATURE_MASK(feature) (1ULL << feature)
#define SMC_DPM_FEATURE ( \ #define SMC_DPM_FEATURE ( \
FEATURE_MASK(FEATURE_DPM_PREFETCHER_BIT) | \ FEATURE_MASK(FEATURE_DPM_PREFETCHER_BIT) | \
...@@ -3441,7 +3440,8 @@ static void sienna_cichlid_dump_pptable(struct smu_context *smu) ...@@ -3441,7 +3440,8 @@ static void sienna_cichlid_dump_pptable(struct smu_context *smu)
static int sienna_cichlid_i2c_xfer(struct i2c_adapter *i2c_adap, static int sienna_cichlid_i2c_xfer(struct i2c_adapter *i2c_adap,
struct i2c_msg *msg, int num_msgs) struct i2c_msg *msg, int num_msgs)
{ {
struct amdgpu_device *adev = to_amdgpu_device(i2c_adap); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(i2c_adap);
struct amdgpu_device *adev = smu_i2c->adev;
struct smu_context *smu = adev->powerplay.pp_handle; struct smu_context *smu = adev->powerplay.pp_handle;
struct smu_table_context *smu_table = &smu->smu_table; struct smu_table_context *smu_table = &smu->smu_table;
struct smu_table *table = &smu_table->driver_table; struct smu_table *table = &smu_table->driver_table;
...@@ -3453,7 +3453,7 @@ static int sienna_cichlid_i2c_xfer(struct i2c_adapter *i2c_adap, ...@@ -3453,7 +3453,7 @@ static int sienna_cichlid_i2c_xfer(struct i2c_adapter *i2c_adap,
if (!req) if (!req)
return -ENOMEM; return -ENOMEM;
req->I2CcontrollerPort = 1; req->I2CcontrollerPort = smu_i2c->port;
req->I2CSpeed = I2C_SPEED_FAST_400K; req->I2CSpeed = I2C_SPEED_FAST_400K;
req->SlaveAddress = msg[0].addr << 1; /* wants an 8-bit address */ req->SlaveAddress = msg[0].addr << 1; /* wants an 8-bit address */
dir = msg[0].flags & I2C_M_RD; dir = msg[0].flags & I2C_M_RD;
...@@ -3531,28 +3531,61 @@ static const struct i2c_adapter_quirks sienna_cichlid_i2c_control_quirks = { ...@@ -3531,28 +3531,61 @@ static const struct i2c_adapter_quirks sienna_cichlid_i2c_control_quirks = {
.max_comb_2nd_msg_len = MAX_SW_I2C_COMMANDS - 2, .max_comb_2nd_msg_len = MAX_SW_I2C_COMMANDS - 2,
}; };
static int sienna_cichlid_i2c_control_init(struct smu_context *smu, struct i2c_adapter *control) static int sienna_cichlid_i2c_control_init(struct smu_context *smu)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_device *adev = smu->adev;
int res; int res, i;
for (i = 0; i < MAX_SMU_I2C_BUSES; i++) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
smu_i2c->adev = adev;
smu_i2c->port = i;
mutex_init(&smu_i2c->mutex);
control->owner = THIS_MODULE; control->owner = THIS_MODULE;
control->class = I2C_CLASS_HWMON; control->class = I2C_CLASS_HWMON;
control->dev.parent = &adev->pdev->dev; control->dev.parent = &adev->pdev->dev;
control->algo = &sienna_cichlid_i2c_algo; control->algo = &sienna_cichlid_i2c_algo;
snprintf(control->name, sizeof(control->name), "AMDGPU SMU"); snprintf(control->name, sizeof(control->name), "AMDGPU SMU %d", i);
control->quirks = &sienna_cichlid_i2c_control_quirks; control->quirks = &sienna_cichlid_i2c_control_quirks;
i2c_set_adapdata(control, smu_i2c);
res = i2c_add_adapter(control); res = i2c_add_adapter(control);
if (res) if (res) {
DRM_ERROR("Failed to register hw i2c, err: %d\n", res); DRM_ERROR("Failed to register hw i2c, err: %d\n", res);
goto Out_err;
}
}
/* assign the buses used for the FRU EEPROM and RAS EEPROM */
/* XXX ideally this would be something in a vbios data table */
adev->pm.ras_eeprom_i2c_bus = &adev->pm.smu_i2c[1].adapter;
adev->pm.fru_eeprom_i2c_bus = &adev->pm.smu_i2c[0].adapter;
return 0;
Out_err:
for ( ; i >= 0; i--) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
i2c_del_adapter(control);
}
return res; return res;
} }
static void sienna_cichlid_i2c_control_fini(struct smu_context *smu, struct i2c_adapter *control) static void sienna_cichlid_i2c_control_fini(struct smu_context *smu)
{ {
struct amdgpu_device *adev = smu->adev;
int i;
for (i = 0; i < MAX_SMU_I2C_BUSES; i++) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
i2c_del_adapter(control); i2c_del_adapter(control);
}
adev->pm.ras_eeprom_i2c_bus = NULL;
adev->pm.fru_eeprom_i2c_bus = NULL;
} }
static ssize_t sienna_cichlid_get_gpu_metrics(struct smu_context *smu, static ssize_t sienna_cichlid_get_gpu_metrics(struct smu_context *smu,
......
...@@ -25,6 +25,7 @@ ...@@ -25,6 +25,7 @@
#include <linux/firmware.h> #include <linux/firmware.h>
#include "amdgpu.h" #include "amdgpu.h"
#include "amdgpu_dpm.h"
#include "amdgpu_smu.h" #include "amdgpu_smu.h"
#include "atomfirmware.h" #include "atomfirmware.h"
#include "amdgpu_atomfirmware.h" #include "amdgpu_atomfirmware.h"
...@@ -56,8 +57,6 @@ ...@@ -56,8 +57,6 @@
#undef pr_info #undef pr_info
#undef pr_debug #undef pr_debug
#define to_amdgpu_device(x) (container_of(x, struct amdgpu_device, pm.smu_i2c))
#define ALDEBARAN_FEA_MAP(smu_feature, aldebaran_feature) \ #define ALDEBARAN_FEA_MAP(smu_feature, aldebaran_feature) \
[smu_feature] = {1, (aldebaran_feature)} [smu_feature] = {1, (aldebaran_feature)}
...@@ -1465,7 +1464,8 @@ static bool aldebaran_is_dpm_running(struct smu_context *smu) ...@@ -1465,7 +1464,8 @@ static bool aldebaran_is_dpm_running(struct smu_context *smu)
static int aldebaran_i2c_xfer(struct i2c_adapter *i2c_adap, static int aldebaran_i2c_xfer(struct i2c_adapter *i2c_adap,
struct i2c_msg *msg, int num_msgs) struct i2c_msg *msg, int num_msgs)
{ {
struct amdgpu_device *adev = to_amdgpu_device(i2c_adap); struct amdgpu_smu_i2c_bus *smu_i2c = i2c_get_adapdata(i2c_adap);
struct amdgpu_device *adev = smu_i2c->adev;
struct smu_context *smu = adev->powerplay.pp_handle; struct smu_context *smu = adev->powerplay.pp_handle;
struct smu_table_context *smu_table = &smu->smu_table; struct smu_table_context *smu_table = &smu->smu_table;
struct smu_table *table = &smu_table->driver_table; struct smu_table *table = &smu_table->driver_table;
...@@ -1477,7 +1477,7 @@ static int aldebaran_i2c_xfer(struct i2c_adapter *i2c_adap, ...@@ -1477,7 +1477,7 @@ static int aldebaran_i2c_xfer(struct i2c_adapter *i2c_adap,
if (!req) if (!req)
return -ENOMEM; return -ENOMEM;
req->I2CcontrollerPort = 0; req->I2CcontrollerPort = smu_i2c->port;
req->I2CSpeed = I2C_SPEED_FAST_400K; req->I2CSpeed = I2C_SPEED_FAST_400K;
req->SlaveAddress = msg[0].addr << 1; /* wants an 8-bit address */ req->SlaveAddress = msg[0].addr << 1; /* wants an 8-bit address */
dir = msg[0].flags & I2C_M_RD; dir = msg[0].flags & I2C_M_RD;
...@@ -1555,28 +1555,52 @@ static const struct i2c_adapter_quirks aldebaran_i2c_control_quirks = { ...@@ -1555,28 +1555,52 @@ static const struct i2c_adapter_quirks aldebaran_i2c_control_quirks = {
.max_comb_2nd_msg_len = MAX_SW_I2C_COMMANDS - 2, .max_comb_2nd_msg_len = MAX_SW_I2C_COMMANDS - 2,
}; };
static int aldebaran_i2c_control_init(struct smu_context *smu, struct i2c_adapter *control) static int aldebaran_i2c_control_init(struct smu_context *smu)
{ {
struct amdgpu_device *adev = to_amdgpu_device(control); struct amdgpu_device *adev = smu->adev;
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[0];
struct i2c_adapter *control = &smu_i2c->adapter;
int res; int res;
smu_i2c->adev = adev;
smu_i2c->port = 0;
mutex_init(&smu_i2c->mutex);
control->owner = THIS_MODULE; control->owner = THIS_MODULE;
control->class = I2C_CLASS_SPD; control->class = I2C_CLASS_SPD;
control->dev.parent = &adev->pdev->dev; control->dev.parent = &adev->pdev->dev;
control->algo = &aldebaran_i2c_algo; control->algo = &aldebaran_i2c_algo;
snprintf(control->name, sizeof(control->name), "AMDGPU SMU"); snprintf(control->name, sizeof(control->name), "AMDGPU SMU 0");
control->quirks = &aldebaran_i2c_control_quirks; control->quirks = &aldebaran_i2c_control_quirks;
i2c_set_adapdata(control, smu_i2c);
res = i2c_add_adapter(control); res = i2c_add_adapter(control);
if (res) if (res) {
DRM_ERROR("Failed to register hw i2c, err: %d\n", res); DRM_ERROR("Failed to register hw i2c, err: %d\n", res);
goto Out_err;
}
adev->pm.ras_eeprom_i2c_bus = &adev->pm.smu_i2c[0].adapter;
return 0;
Out_err:
i2c_del_adapter(control);
return res; return res;
} }
static void aldebaran_i2c_control_fini(struct smu_context *smu, struct i2c_adapter *control) static void aldebaran_i2c_control_fini(struct smu_context *smu)
{ {
struct amdgpu_device *adev = smu->adev;
int i;
for (i = 0; i < MAX_SMU_I2C_BUSES; i++) {
struct amdgpu_smu_i2c_bus *smu_i2c = &adev->pm.smu_i2c[i];
struct i2c_adapter *control = &smu_i2c->adapter;
i2c_del_adapter(control); i2c_del_adapter(control);
}
adev->pm.ras_eeprom_i2c_bus = NULL;
adev->pm.fru_eeprom_i2c_bus = NULL;
} }
static void aldebaran_get_unique_id(struct smu_context *smu) static void aldebaran_get_unique_id(struct smu_context *smu)
......
...@@ -78,8 +78,8 @@ ...@@ -78,8 +78,8 @@
#define smu_dump_pptable(smu) smu_ppt_funcs(dump_pptable, 0, smu) #define smu_dump_pptable(smu) smu_ppt_funcs(dump_pptable, 0, smu)
#define smu_update_pcie_parameters(smu, pcie_gen_cap, pcie_width_cap) smu_ppt_funcs(update_pcie_parameters, 0, smu, pcie_gen_cap, pcie_width_cap) #define smu_update_pcie_parameters(smu, pcie_gen_cap, pcie_width_cap) smu_ppt_funcs(update_pcie_parameters, 0, smu, pcie_gen_cap, pcie_width_cap)
#define smu_set_power_source(smu, power_src) smu_ppt_funcs(set_power_source, 0, smu, power_src) #define smu_set_power_source(smu, power_src) smu_ppt_funcs(set_power_source, 0, smu, power_src)
#define smu_i2c_init(smu, control) smu_ppt_funcs(i2c_init, 0, smu, control) #define smu_i2c_init(smu) smu_ppt_funcs(i2c_init, 0, smu)
#define smu_i2c_fini(smu, control) smu_ppt_funcs(i2c_fini, 0, smu, control) #define smu_i2c_fini(smu) smu_ppt_funcs(i2c_fini, 0, smu)
#define smu_get_unique_id(smu) smu_ppt_funcs(get_unique_id, 0, smu) #define smu_get_unique_id(smu) smu_ppt_funcs(get_unique_id, 0, smu)
#define smu_log_thermal_throttling(smu) smu_ppt_funcs(log_thermal_throttling_event, 0, smu) #define smu_log_thermal_throttling(smu) smu_ppt_funcs(log_thermal_throttling_event, 0, smu)
#define smu_get_asic_power_limits(smu, current, default, max) smu_ppt_funcs(get_power_limit, 0, smu, current, default, max) #define smu_get_asic_power_limits(smu, current, default, max) smu_ppt_funcs(get_power_limit, 0, smu, current, default, max)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment