486 lines
11 KiB
C
486 lines
11 KiB
C
|
/*
|
||
|
* Copyright 2012-15 Advanced Micro Devices, Inc.
|
||
|
*
|
||
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
||
|
* copy of this software and associated documentation files (the "Software"),
|
||
|
* to deal in the Software without restriction, including without limitation
|
||
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||
|
* and/or sell copies of the Software, and to permit persons to whom the
|
||
|
* Software is furnished to do so, subject to the following conditions:
|
||
|
*
|
||
|
* The above copyright notice and this permission notice shall be included in
|
||
|
* all copies or substantial portions of the Software.
|
||
|
*
|
||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||
|
* THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
||
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
||
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||
|
* OTHER DEALINGS IN THE SOFTWARE.
|
||
|
*
|
||
|
* Authors: AMD
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include "dm_services.h"
|
||
|
|
||
|
/*
|
||
|
* Pre-requisites: headers required by header of this unit
|
||
|
*/
|
||
|
#include "include/i2caux_interface.h"
|
||
|
#include "dc_bios_types.h"
|
||
|
|
||
|
/*
|
||
|
* Header of this unit
|
||
|
*/
|
||
|
|
||
|
#include "i2caux.h"
|
||
|
|
||
|
/*
|
||
|
* Post-requisites: headers required by this unit
|
||
|
*/
|
||
|
|
||
|
#include "engine.h"
|
||
|
#include "i2c_engine.h"
|
||
|
#include "aux_engine.h"
|
||
|
|
||
|
/*
|
||
|
* This unit
|
||
|
*/
|
||
|
|
||
|
#include "dce80/i2caux_dce80.h"
|
||
|
|
||
|
#include "dce100/i2caux_dce100.h"
|
||
|
|
||
|
#include "dce110/i2caux_dce110.h"
|
||
|
|
||
|
#include "dce112/i2caux_dce112.h"
|
||
|
|
||
|
#include "dce120/i2caux_dce120.h"
|
||
|
|
||
|
#if defined(CONFIG_DRM_AMD_DC_DCN1_0)
|
||
|
#include "dcn10/i2caux_dcn10.h"
|
||
|
#endif
|
||
|
|
||
|
#include "diagnostics/i2caux_diag.h"
|
||
|
|
||
|
/*
|
||
|
* @brief
|
||
|
* Plain API, available publicly
|
||
|
*/
|
||
|
|
||
|
struct i2caux *dal_i2caux_create(
|
||
|
struct dc_context *ctx)
|
||
|
{
|
||
|
if (IS_FPGA_MAXIMUS_DC(ctx->dce_environment)) {
|
||
|
return dal_i2caux_diag_fpga_create(ctx);
|
||
|
}
|
||
|
|
||
|
switch (ctx->dce_version) {
|
||
|
case DCE_VERSION_8_0:
|
||
|
case DCE_VERSION_8_1:
|
||
|
case DCE_VERSION_8_3:
|
||
|
return dal_i2caux_dce80_create(ctx);
|
||
|
case DCE_VERSION_11_2:
|
||
|
return dal_i2caux_dce112_create(ctx);
|
||
|
case DCE_VERSION_11_0:
|
||
|
return dal_i2caux_dce110_create(ctx);
|
||
|
case DCE_VERSION_10_0:
|
||
|
return dal_i2caux_dce100_create(ctx);
|
||
|
case DCE_VERSION_12_0:
|
||
|
return dal_i2caux_dce120_create(ctx);
|
||
|
#if defined(CONFIG_DRM_AMD_DC_DCN1_0)
|
||
|
case DCN_VERSION_1_0:
|
||
|
return dal_i2caux_dcn10_create(ctx);
|
||
|
#endif
|
||
|
|
||
|
default:
|
||
|
BREAK_TO_DEBUGGER();
|
||
|
return NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool dal_i2caux_submit_i2c_command(
|
||
|
struct i2caux *i2caux,
|
||
|
struct ddc *ddc,
|
||
|
struct i2c_command *cmd)
|
||
|
{
|
||
|
struct i2c_engine *engine;
|
||
|
uint8_t index_of_payload = 0;
|
||
|
bool result;
|
||
|
|
||
|
if (!ddc) {
|
||
|
BREAK_TO_DEBUGGER();
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if (!cmd) {
|
||
|
BREAK_TO_DEBUGGER();
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* default will be SW, however there is a feature flag in adapter
|
||
|
* service that determines whether SW i2c_engine will be available or
|
||
|
* not, if sw i2c is not available we will fallback to hw. This feature
|
||
|
* flag is set to not creating sw i2c engine for every dce except dce80
|
||
|
* currently
|
||
|
*/
|
||
|
switch (cmd->engine) {
|
||
|
case I2C_COMMAND_ENGINE_DEFAULT:
|
||
|
case I2C_COMMAND_ENGINE_SW:
|
||
|
/* try to acquire SW engine first,
|
||
|
* acquire HW engine if SW engine not available */
|
||
|
engine = i2caux->funcs->acquire_i2c_sw_engine(i2caux, ddc);
|
||
|
|
||
|
if (!engine)
|
||
|
engine = i2caux->funcs->acquire_i2c_hw_engine(
|
||
|
i2caux, ddc);
|
||
|
break;
|
||
|
case I2C_COMMAND_ENGINE_HW:
|
||
|
default:
|
||
|
/* try to acquire HW engine first,
|
||
|
* acquire SW engine if HW engine not available */
|
||
|
engine = i2caux->funcs->acquire_i2c_hw_engine(i2caux, ddc);
|
||
|
|
||
|
if (!engine)
|
||
|
engine = i2caux->funcs->acquire_i2c_sw_engine(
|
||
|
i2caux, ddc);
|
||
|
}
|
||
|
|
||
|
if (!engine)
|
||
|
return false;
|
||
|
|
||
|
engine->funcs->set_speed(engine, cmd->speed);
|
||
|
|
||
|
result = true;
|
||
|
|
||
|
while (index_of_payload < cmd->number_of_payloads) {
|
||
|
bool mot = (index_of_payload != cmd->number_of_payloads - 1);
|
||
|
|
||
|
struct i2c_payload *payload = cmd->payloads + index_of_payload;
|
||
|
|
||
|
struct i2caux_transaction_request request = { 0 };
|
||
|
|
||
|
request.operation = payload->write ?
|
||
|
I2CAUX_TRANSACTION_WRITE :
|
||
|
I2CAUX_TRANSACTION_READ;
|
||
|
|
||
|
request.payload.address_space =
|
||
|
I2CAUX_TRANSACTION_ADDRESS_SPACE_I2C;
|
||
|
request.payload.address = (payload->address << 1) |
|
||
|
!payload->write;
|
||
|
request.payload.length = payload->length;
|
||
|
request.payload.data = payload->data;
|
||
|
|
||
|
if (!engine->base.funcs->submit_request(
|
||
|
&engine->base, &request, mot)) {
|
||
|
result = false;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
++index_of_payload;
|
||
|
}
|
||
|
|
||
|
i2caux->funcs->release_engine(i2caux, &engine->base);
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
bool dal_i2caux_submit_aux_command(
|
||
|
struct i2caux *i2caux,
|
||
|
struct ddc *ddc,
|
||
|
struct aux_command *cmd)
|
||
|
{
|
||
|
struct aux_engine *engine;
|
||
|
uint8_t index_of_payload = 0;
|
||
|
bool result;
|
||
|
bool mot;
|
||
|
|
||
|
if (!ddc) {
|
||
|
BREAK_TO_DEBUGGER();
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if (!cmd) {
|
||
|
BREAK_TO_DEBUGGER();
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
engine = i2caux->funcs->acquire_aux_engine(i2caux, ddc);
|
||
|
|
||
|
if (!engine)
|
||
|
return false;
|
||
|
|
||
|
engine->delay = cmd->defer_delay;
|
||
|
engine->max_defer_write_retry = cmd->max_defer_write_retry;
|
||
|
|
||
|
result = true;
|
||
|
|
||
|
while (index_of_payload < cmd->number_of_payloads) {
|
||
|
struct aux_payload *payload = cmd->payloads + index_of_payload;
|
||
|
struct i2caux_transaction_request request = { 0 };
|
||
|
|
||
|
if (cmd->mot == I2C_MOT_UNDEF)
|
||
|
mot = (index_of_payload != cmd->number_of_payloads - 1);
|
||
|
else
|
||
|
mot = (cmd->mot == I2C_MOT_TRUE);
|
||
|
|
||
|
request.operation = payload->write ?
|
||
|
I2CAUX_TRANSACTION_WRITE :
|
||
|
I2CAUX_TRANSACTION_READ;
|
||
|
|
||
|
if (payload->i2c_over_aux) {
|
||
|
request.payload.address_space =
|
||
|
I2CAUX_TRANSACTION_ADDRESS_SPACE_I2C;
|
||
|
|
||
|
request.payload.address = (payload->address << 1) |
|
||
|
!payload->write;
|
||
|
} else {
|
||
|
request.payload.address_space =
|
||
|
I2CAUX_TRANSACTION_ADDRESS_SPACE_DPCD;
|
||
|
|
||
|
request.payload.address = payload->address;
|
||
|
}
|
||
|
|
||
|
request.payload.length = payload->length;
|
||
|
request.payload.data = payload->data;
|
||
|
|
||
|
if (!engine->base.funcs->submit_request(
|
||
|
&engine->base, &request, mot)) {
|
||
|
result = false;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
++index_of_payload;
|
||
|
}
|
||
|
|
||
|
i2caux->funcs->release_engine(i2caux, &engine->base);
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
static bool get_hw_supported_ddc_line(
|
||
|
struct ddc *ddc,
|
||
|
enum gpio_ddc_line *line)
|
||
|
{
|
||
|
enum gpio_ddc_line line_found;
|
||
|
|
||
|
*line = GPIO_DDC_LINE_UNKNOWN;
|
||
|
|
||
|
if (!ddc) {
|
||
|
BREAK_TO_DEBUGGER();
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if (!ddc->hw_info.hw_supported)
|
||
|
return false;
|
||
|
|
||
|
line_found = dal_ddc_get_line(ddc);
|
||
|
|
||
|
if (line_found >= GPIO_DDC_LINE_COUNT)
|
||
|
return false;
|
||
|
|
||
|
*line = line_found;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void dal_i2caux_configure_aux(
|
||
|
struct i2caux *i2caux,
|
||
|
struct ddc *ddc,
|
||
|
union aux_config cfg)
|
||
|
{
|
||
|
struct aux_engine *engine =
|
||
|
i2caux->funcs->acquire_aux_engine(i2caux, ddc);
|
||
|
|
||
|
if (!engine)
|
||
|
return;
|
||
|
|
||
|
engine->funcs->configure(engine, cfg);
|
||
|
|
||
|
i2caux->funcs->release_engine(i2caux, &engine->base);
|
||
|
}
|
||
|
|
||
|
void dal_i2caux_destroy(
|
||
|
struct i2caux **i2caux)
|
||
|
{
|
||
|
if (!i2caux || !*i2caux) {
|
||
|
BREAK_TO_DEBUGGER();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
(*i2caux)->funcs->destroy(i2caux);
|
||
|
|
||
|
*i2caux = NULL;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* @brief
|
||
|
* An utility function used by 'struct i2caux' and its descendants
|
||
|
*/
|
||
|
|
||
|
uint32_t dal_i2caux_get_reference_clock(
|
||
|
struct dc_bios *bios)
|
||
|
{
|
||
|
struct dc_firmware_info info = { { 0 } };
|
||
|
|
||
|
if (bios->funcs->get_firmware_info(bios, &info) != BP_RESULT_OK)
|
||
|
return 0;
|
||
|
|
||
|
return info.pll_info.crystal_frequency;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* @brief
|
||
|
* i2caux
|
||
|
*/
|
||
|
|
||
|
enum {
|
||
|
/* following are expressed in KHz */
|
||
|
DEFAULT_I2C_SW_SPEED = 50,
|
||
|
DEFAULT_I2C_HW_SPEED = 50,
|
||
|
|
||
|
DEFAULT_I2C_SW_SPEED_100KHZ = 100,
|
||
|
DEFAULT_I2C_HW_SPEED_100KHZ = 100,
|
||
|
|
||
|
/* This is the timeout as defined in DP 1.2a,
|
||
|
* 2.3.4 "Detailed uPacket TX AUX CH State Description". */
|
||
|
AUX_TIMEOUT_PERIOD = 400,
|
||
|
|
||
|
/* Ideally, the SW timeout should be just above 550usec
|
||
|
* which is programmed in HW.
|
||
|
* But the SW timeout of 600usec is not reliable,
|
||
|
* because on some systems, delay_in_microseconds()
|
||
|
* returns faster than it should.
|
||
|
* EPR #379763: by trial-and-error on different systems,
|
||
|
* 700usec is the minimum reliable SW timeout for polling
|
||
|
* the AUX_SW_STATUS.AUX_SW_DONE bit.
|
||
|
* This timeout expires *only* when there is
|
||
|
* AUX Error or AUX Timeout conditions - not during normal operation.
|
||
|
* During normal operation, AUX_SW_STATUS.AUX_SW_DONE bit is set
|
||
|
* at most within ~240usec. That means,
|
||
|
* increasing this timeout will not affect normal operation,
|
||
|
* and we'll timeout after
|
||
|
* SW_AUX_TIMEOUT_PERIOD_MULTIPLIER * AUX_TIMEOUT_PERIOD = 1600usec.
|
||
|
* This timeout is especially important for
|
||
|
* resume from S3 and CTS. */
|
||
|
SW_AUX_TIMEOUT_PERIOD_MULTIPLIER = 4
|
||
|
};
|
||
|
|
||
|
struct i2c_engine *dal_i2caux_acquire_i2c_sw_engine(
|
||
|
struct i2caux *i2caux,
|
||
|
struct ddc *ddc)
|
||
|
{
|
||
|
enum gpio_ddc_line line;
|
||
|
struct i2c_engine *engine = NULL;
|
||
|
|
||
|
if (get_hw_supported_ddc_line(ddc, &line))
|
||
|
engine = i2caux->i2c_sw_engines[line];
|
||
|
|
||
|
if (!engine)
|
||
|
engine = i2caux->i2c_generic_sw_engine;
|
||
|
|
||
|
if (!engine)
|
||
|
return NULL;
|
||
|
|
||
|
if (!engine->base.funcs->acquire(&engine->base, ddc))
|
||
|
return NULL;
|
||
|
|
||
|
return engine;
|
||
|
}
|
||
|
|
||
|
struct aux_engine *dal_i2caux_acquire_aux_engine(
|
||
|
struct i2caux *i2caux,
|
||
|
struct ddc *ddc)
|
||
|
{
|
||
|
enum gpio_ddc_line line;
|
||
|
struct aux_engine *engine;
|
||
|
|
||
|
if (!get_hw_supported_ddc_line(ddc, &line))
|
||
|
return NULL;
|
||
|
|
||
|
engine = i2caux->aux_engines[line];
|
||
|
|
||
|
if (!engine)
|
||
|
return NULL;
|
||
|
|
||
|
if (!engine->base.funcs->acquire(&engine->base, ddc))
|
||
|
return NULL;
|
||
|
|
||
|
return engine;
|
||
|
}
|
||
|
|
||
|
void dal_i2caux_release_engine(
|
||
|
struct i2caux *i2caux,
|
||
|
struct engine *engine)
|
||
|
{
|
||
|
engine->funcs->release_engine(engine);
|
||
|
|
||
|
dal_ddc_close(engine->ddc);
|
||
|
|
||
|
engine->ddc = NULL;
|
||
|
}
|
||
|
|
||
|
void dal_i2caux_construct(
|
||
|
struct i2caux *i2caux,
|
||
|
struct dc_context *ctx)
|
||
|
{
|
||
|
uint32_t i = 0;
|
||
|
|
||
|
i2caux->ctx = ctx;
|
||
|
do {
|
||
|
i2caux->i2c_sw_engines[i] = NULL;
|
||
|
i2caux->i2c_hw_engines[i] = NULL;
|
||
|
i2caux->aux_engines[i] = NULL;
|
||
|
|
||
|
++i;
|
||
|
} while (i < GPIO_DDC_LINE_COUNT);
|
||
|
|
||
|
i2caux->i2c_generic_sw_engine = NULL;
|
||
|
i2caux->i2c_generic_hw_engine = NULL;
|
||
|
|
||
|
i2caux->aux_timeout_period =
|
||
|
SW_AUX_TIMEOUT_PERIOD_MULTIPLIER * AUX_TIMEOUT_PERIOD;
|
||
|
|
||
|
if (ctx->dce_version >= DCE_VERSION_11_2) {
|
||
|
i2caux->default_i2c_hw_speed = DEFAULT_I2C_HW_SPEED_100KHZ;
|
||
|
i2caux->default_i2c_sw_speed = DEFAULT_I2C_SW_SPEED_100KHZ;
|
||
|
} else {
|
||
|
i2caux->default_i2c_hw_speed = DEFAULT_I2C_HW_SPEED;
|
||
|
i2caux->default_i2c_sw_speed = DEFAULT_I2C_SW_SPEED;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void dal_i2caux_destruct(
|
||
|
struct i2caux *i2caux)
|
||
|
{
|
||
|
uint32_t i = 0;
|
||
|
|
||
|
if (i2caux->i2c_generic_hw_engine)
|
||
|
i2caux->i2c_generic_hw_engine->funcs->destroy(
|
||
|
&i2caux->i2c_generic_hw_engine);
|
||
|
|
||
|
if (i2caux->i2c_generic_sw_engine)
|
||
|
i2caux->i2c_generic_sw_engine->funcs->destroy(
|
||
|
&i2caux->i2c_generic_sw_engine);
|
||
|
|
||
|
do {
|
||
|
if (i2caux->aux_engines[i])
|
||
|
i2caux->aux_engines[i]->funcs->destroy(
|
||
|
&i2caux->aux_engines[i]);
|
||
|
|
||
|
if (i2caux->i2c_hw_engines[i])
|
||
|
i2caux->i2c_hw_engines[i]->funcs->destroy(
|
||
|
&i2caux->i2c_hw_engines[i]);
|
||
|
|
||
|
if (i2caux->i2c_sw_engines[i])
|
||
|
i2caux->i2c_sw_engines[i]->funcs->destroy(
|
||
|
&i2caux->i2c_sw_engines[i]);
|
||
|
|
||
|
++i;
|
||
|
} while (i < GPIO_DDC_LINE_COUNT);
|
||
|
}
|
||
|
|