Release 4.17 drivers/gpu/drm/amd/display/dc/i2caux/i2caux.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;
	}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 80 | 71.43% | 1 | 25.00% | 
| Alex Deucher | 32 | 28.57% | 3 | 75.00% | 
| Total | 112 | 100.00% | 4 | 100.00% | 
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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 318 | 100.00% | 1 | 100.00% | 
| Total | 318 | 100.00% | 1 | 100.00% | 
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;
		}
		cmd->payloads->length = request.payload.length;
		++index_of_payload;
	}
	i2caux->funcs->release_engine(i2caux, &engine->base);
	return result;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 266 | 85.26% | 1 | 33.33% | 
| Andrey Grodzovsky | 34 | 10.90% | 1 | 33.33% | 
| Shirish S | 12 | 3.85% | 1 | 33.33% | 
| Total | 312 | 100.00% | 3 | 100.00% | 
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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 74 | 100.00% | 2 | 100.00% | 
| Total | 74 | 100.00% | 2 | 100.00% | 
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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 66 | 100.00% | 1 | 100.00% | 
| Total | 66 | 100.00% | 1 | 100.00% | 
void dal_i2caux_destroy(
	struct i2caux **i2caux)
{
	if (!i2caux || !*i2caux) {
		BREAK_TO_DEBUGGER();
		return;
	}
	(*i2caux)->funcs->destroy(i2caux);
	*i2caux = NULL;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 43 | 100.00% | 1 | 100.00% | 
| Total | 43 | 100.00% | 1 | 100.00% | 
/*
 * @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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 46 | 100.00% | 2 | 100.00% | 
| Total | 46 | 100.00% | 2 | 100.00% | 
/*
 * @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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 91 | 100.00% | 1 | 100.00% | 
| Total | 91 | 100.00% | 1 | 100.00% | 
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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 82 | 100.00% | 1 | 100.00% | 
| Total | 82 | 100.00% | 1 | 100.00% | 
void dal_i2caux_release_engine(
	struct i2caux *i2caux,
	struct engine *engine)
{
	engine->funcs->release_engine(engine);
	dal_ddc_close(engine->ddc);
	engine->ddc = NULL;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 37 | 100.00% | 1 | 100.00% | 
| Total | 37 | 100.00% | 1 | 100.00% | 
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;
	}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 96 | 78.05% | 1 | 33.33% | 
| Zeyu Fan | 26 | 21.14% | 1 | 33.33% | 
| Dave Airlie | 1 | 0.81% | 1 | 33.33% | 
| Total | 123 | 100.00% | 3 | 100.00% | 
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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 155 | 100.00% | 1 | 100.00% | 
| Total | 155 | 100.00% | 1 | 100.00% | 
Overall Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Harry Wentland | 1420 | 91.79% | 3 | 30.00% | 
| Alex Deucher | 46 | 2.97% | 3 | 30.00% | 
| Zeyu Fan | 34 | 2.20% | 1 | 10.00% | 
| Andrey Grodzovsky | 34 | 2.20% | 1 | 10.00% | 
| Shirish S | 12 | 0.78% | 1 | 10.00% | 
| Dave Airlie | 1 | 0.06% | 1 | 10.00% | 
| Total | 1547 | 100.00% | 10 | 100.00% | 
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.