cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

dmub_abm.c (8392B)


      1/*
      2 * Copyright 2019 Advanced Micro Devices, Inc.
      3 *
      4 * Permission is hereby granted, free of charge, to any person obtaining a
      5 * copy of this software and associated documentation files (the "Software"),
      6 * to deal in the Software without restriction, including without limitation
      7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
      8 * and/or sell copies of the Software, and to permit persons to whom the
      9 * Software is furnished to do so, subject to the following conditions:
     10 *
     11 * The above copyright notice and this permission notice shall be included in
     12 * all copies or substantial portions of the Software.
     13 *
     14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
     15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
     16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
     17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
     18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
     19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
     20 * OTHER DEALINGS IN THE SOFTWARE.
     21 *
     22 * Authors: AMD
     23 *
     24 */
     25
     26#include "dmub_abm.h"
     27#include "dce_abm.h"
     28#include "dc.h"
     29#include "dc_dmub_srv.h"
     30#include "dmub/dmub_srv.h"
     31#include "core_types.h"
     32#include "dm_services.h"
     33#include "reg_helper.h"
     34#include "fixed31_32.h"
     35
     36#include "atom.h"
     37
     38#define TO_DMUB_ABM(abm)\
     39	container_of(abm, struct dce_abm, base)
     40
     41#define REG(reg) \
     42	(dce_abm->regs->reg)
     43
     44#undef FN
     45#define FN(reg_name, field_name) \
     46	dce_abm->abm_shift->field_name, dce_abm->abm_mask->field_name
     47
     48#define CTX \
     49	dce_abm->base.ctx
     50
     51#define DISABLE_ABM_IMMEDIATELY 255
     52
     53
     54
     55static void dmub_abm_enable_fractional_pwm(struct dc_context *dc)
     56{
     57	union dmub_rb_cmd cmd;
     58	uint32_t fractional_pwm = (dc->dc->config.disable_fractional_pwm == false) ? 1 : 0;
     59	uint32_t edp_id_count = dc->dc_edp_id_count;
     60	int i;
     61	uint8_t panel_mask = 0;
     62
     63	for (i = 0; i < edp_id_count; i++)
     64		panel_mask |= 0x01 << i;
     65
     66	memset(&cmd, 0, sizeof(cmd));
     67	cmd.abm_set_pwm_frac.header.type = DMUB_CMD__ABM;
     68	cmd.abm_set_pwm_frac.header.sub_type = DMUB_CMD__ABM_SET_PWM_FRAC;
     69	cmd.abm_set_pwm_frac.abm_set_pwm_frac_data.fractional_pwm = fractional_pwm;
     70	cmd.abm_set_pwm_frac.abm_set_pwm_frac_data.version = DMUB_CMD_ABM_CONTROL_VERSION_1;
     71	cmd.abm_set_pwm_frac.abm_set_pwm_frac_data.panel_mask = panel_mask;
     72	cmd.abm_set_pwm_frac.header.payload_bytes = sizeof(struct dmub_cmd_abm_set_pwm_frac_data);
     73
     74	dc_dmub_srv_cmd_queue(dc->dmub_srv, &cmd);
     75	dc_dmub_srv_cmd_execute(dc->dmub_srv);
     76	dc_dmub_srv_wait_idle(dc->dmub_srv);
     77}
     78
     79static void dmub_abm_init(struct abm *abm, uint32_t backlight)
     80{
     81	struct dce_abm *dce_abm = TO_DMUB_ABM(abm);
     82
     83	REG_WRITE(DC_ABM1_HG_SAMPLE_RATE, 0x3);
     84	REG_WRITE(DC_ABM1_HG_SAMPLE_RATE, 0x1);
     85	REG_WRITE(DC_ABM1_LS_SAMPLE_RATE, 0x3);
     86	REG_WRITE(DC_ABM1_LS_SAMPLE_RATE, 0x1);
     87	REG_WRITE(BL1_PWM_BL_UPDATE_SAMPLE_RATE, 0x1);
     88
     89	REG_SET_3(DC_ABM1_HG_MISC_CTRL, 0,
     90			ABM1_HG_NUM_OF_BINS_SEL, 0,
     91			ABM1_HG_VMAX_SEL, 1,
     92			ABM1_HG_BIN_BITWIDTH_SIZE_SEL, 0);
     93
     94	REG_SET_3(DC_ABM1_IPCSC_COEFF_SEL, 0,
     95			ABM1_IPCSC_COEFF_SEL_R, 2,
     96			ABM1_IPCSC_COEFF_SEL_G, 4,
     97			ABM1_IPCSC_COEFF_SEL_B, 2);
     98
     99	REG_UPDATE(BL1_PWM_CURRENT_ABM_LEVEL,
    100			BL1_PWM_CURRENT_ABM_LEVEL, backlight);
    101
    102	REG_UPDATE(BL1_PWM_TARGET_ABM_LEVEL,
    103			BL1_PWM_TARGET_ABM_LEVEL, backlight);
    104
    105	REG_UPDATE(BL1_PWM_USER_LEVEL,
    106			BL1_PWM_USER_LEVEL, backlight);
    107
    108	REG_UPDATE_2(DC_ABM1_LS_MIN_MAX_PIXEL_VALUE_THRES,
    109			ABM1_LS_MIN_PIXEL_VALUE_THRES, 0,
    110			ABM1_LS_MAX_PIXEL_VALUE_THRES, 1000);
    111
    112	REG_SET_3(DC_ABM1_HGLS_REG_READ_PROGRESS, 0,
    113			ABM1_HG_REG_READ_MISSED_FRAME_CLEAR, 1,
    114			ABM1_LS_REG_READ_MISSED_FRAME_CLEAR, 1,
    115			ABM1_BL_REG_READ_MISSED_FRAME_CLEAR, 1);
    116
    117	dmub_abm_enable_fractional_pwm(abm->ctx);
    118}
    119
    120static unsigned int dmub_abm_get_current_backlight(struct abm *abm)
    121{
    122	struct dce_abm *dce_abm = TO_DMUB_ABM(abm);
    123	unsigned int backlight = REG_READ(BL1_PWM_CURRENT_ABM_LEVEL);
    124
    125	/* return backlight in hardware format which is unsigned 17 bits, with
    126	 * 1 bit integer and 16 bit fractional
    127	 */
    128	return backlight;
    129}
    130
    131static unsigned int dmub_abm_get_target_backlight(struct abm *abm)
    132{
    133	struct dce_abm *dce_abm = TO_DMUB_ABM(abm);
    134	unsigned int backlight = REG_READ(BL1_PWM_TARGET_ABM_LEVEL);
    135
    136	/* return backlight in hardware format which is unsigned 17 bits, with
    137	 * 1 bit integer and 16 bit fractional
    138	 */
    139	return backlight;
    140}
    141
    142static bool dmub_abm_set_level(struct abm *abm, uint32_t level)
    143{
    144	union dmub_rb_cmd cmd;
    145	struct dc_context *dc = abm->ctx;
    146	struct dc_link *edp_links[MAX_NUM_EDP];
    147	int i;
    148	int edp_num;
    149	uint8_t panel_mask = 0;
    150
    151	get_edp_links(dc->dc, edp_links, &edp_num);
    152
    153	for (i = 0; i < edp_num; i++) {
    154		if (edp_links[i]->link_status.link_active)
    155			panel_mask |= (0x01 << i);
    156	}
    157
    158	memset(&cmd, 0, sizeof(cmd));
    159	cmd.abm_set_level.header.type = DMUB_CMD__ABM;
    160	cmd.abm_set_level.header.sub_type = DMUB_CMD__ABM_SET_LEVEL;
    161	cmd.abm_set_level.abm_set_level_data.level = level;
    162	cmd.abm_set_level.abm_set_level_data.version = DMUB_CMD_ABM_CONTROL_VERSION_1;
    163	cmd.abm_set_level.abm_set_level_data.panel_mask = panel_mask;
    164	cmd.abm_set_level.header.payload_bytes = sizeof(struct dmub_cmd_abm_set_level_data);
    165
    166	dc_dmub_srv_cmd_queue(dc->dmub_srv, &cmd);
    167	dc_dmub_srv_cmd_execute(dc->dmub_srv);
    168	dc_dmub_srv_wait_idle(dc->dmub_srv);
    169
    170	return true;
    171}
    172
    173static bool dmub_abm_init_config(struct abm *abm,
    174	const char *src,
    175	unsigned int bytes,
    176	unsigned int inst)
    177{
    178	union dmub_rb_cmd cmd;
    179	struct dc_context *dc = abm->ctx;
    180	uint8_t panel_mask = 0x01 << inst;
    181
    182	// TODO: Optimize by only reading back final 4 bytes
    183	dmub_flush_buffer_mem(&dc->dmub_srv->dmub->scratch_mem_fb);
    184
    185	// Copy iramtable into cw7
    186	memcpy(dc->dmub_srv->dmub->scratch_mem_fb.cpu_addr, (void *)src, bytes);
    187
    188	memset(&cmd, 0, sizeof(cmd));
    189	// Fw will copy from cw7 to fw_state
    190	cmd.abm_init_config.header.type = DMUB_CMD__ABM;
    191	cmd.abm_init_config.header.sub_type = DMUB_CMD__ABM_INIT_CONFIG;
    192	cmd.abm_init_config.abm_init_config_data.src.quad_part = dc->dmub_srv->dmub->scratch_mem_fb.gpu_addr;
    193	cmd.abm_init_config.abm_init_config_data.bytes = bytes;
    194	cmd.abm_init_config.abm_init_config_data.version = DMUB_CMD_ABM_CONTROL_VERSION_1;
    195	cmd.abm_init_config.abm_init_config_data.panel_mask = panel_mask;
    196
    197	cmd.abm_init_config.header.payload_bytes = sizeof(struct dmub_cmd_abm_init_config_data);
    198
    199	dc_dmub_srv_cmd_queue(dc->dmub_srv, &cmd);
    200	dc_dmub_srv_cmd_execute(dc->dmub_srv);
    201	dc_dmub_srv_wait_idle(dc->dmub_srv);
    202
    203	return true;
    204}
    205
    206static bool dmub_abm_set_pause(struct abm *abm, bool pause, unsigned int panel_inst, unsigned int stream_inst)
    207{
    208	union dmub_rb_cmd cmd;
    209	struct dc_context *dc = abm->ctx;
    210	uint8_t panel_mask = 0x01 << panel_inst;
    211
    212	memset(&cmd, 0, sizeof(cmd));
    213	cmd.abm_pause.header.type = DMUB_CMD__ABM;
    214	cmd.abm_pause.header.sub_type = DMUB_CMD__ABM_PAUSE;
    215	cmd.abm_pause.abm_pause_data.enable = pause;
    216	cmd.abm_pause.abm_pause_data.panel_mask = panel_mask;
    217	cmd.abm_set_level.header.payload_bytes = sizeof(struct dmub_cmd_abm_pause_data);
    218
    219	dc_dmub_srv_cmd_queue(dc->dmub_srv, &cmd);
    220	dc_dmub_srv_cmd_execute(dc->dmub_srv);
    221	dc_dmub_srv_wait_idle(dc->dmub_srv);
    222
    223	return true;
    224}
    225
    226static const struct abm_funcs abm_funcs = {
    227	.abm_init = dmub_abm_init,
    228	.set_abm_level = dmub_abm_set_level,
    229	.get_current_backlight = dmub_abm_get_current_backlight,
    230	.get_target_backlight = dmub_abm_get_target_backlight,
    231	.init_abm_config = dmub_abm_init_config,
    232	.set_abm_pause = dmub_abm_set_pause,
    233};
    234
    235static void dmub_abm_construct(
    236	struct dce_abm *abm_dce,
    237	struct dc_context *ctx,
    238	const struct dce_abm_registers *regs,
    239	const struct dce_abm_shift *abm_shift,
    240	const struct dce_abm_mask *abm_mask)
    241{
    242	struct abm *base = &abm_dce->base;
    243
    244	base->ctx = ctx;
    245	base->funcs = &abm_funcs;
    246	base->dmcu_is_running = false;
    247
    248	abm_dce->regs = regs;
    249	abm_dce->abm_shift = abm_shift;
    250	abm_dce->abm_mask = abm_mask;
    251}
    252
    253struct abm *dmub_abm_create(
    254	struct dc_context *ctx,
    255	const struct dce_abm_registers *regs,
    256	const struct dce_abm_shift *abm_shift,
    257	const struct dce_abm_mask *abm_mask)
    258{
    259	struct dce_abm *abm_dce = kzalloc(sizeof(*abm_dce), GFP_KERNEL);
    260
    261	if (abm_dce == NULL) {
    262		BREAK_TO_DEBUGGER();
    263		return NULL;
    264	}
    265
    266	dmub_abm_construct(abm_dce, ctx, regs, abm_shift, abm_mask);
    267
    268	return &abm_dce->base;
    269}
    270
    271void dmub_abm_destroy(struct abm **abm)
    272{
    273	struct dce_abm *abm_dce = TO_DMUB_ABM(*abm);
    274
    275	kfree(abm_dce);
    276	*abm = NULL;
    277}