208 lines
5.3 KiB
C
208 lines
5.3 KiB
C
/*
|
|
* Copyright 2019 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 "dmub_abm.h"
|
|
#include "dmub_abm_lcd.h"
|
|
#include "dc.h"
|
|
#include "core_types.h"
|
|
|
|
#define TO_DMUB_ABM(abm)\
|
|
container_of(abm, struct dce_abm, base)
|
|
|
|
#define ABM_FEATURE_NO_SUPPORT 0
|
|
#define ABM_LCD_SUPPORT 1
|
|
|
|
static unsigned int abm_feature_support(struct abm *abm, unsigned int panel_inst)
|
|
{
|
|
struct dc_context *dc = abm->ctx;
|
|
struct dc_link *edp_links[MAX_NUM_EDP];
|
|
int i;
|
|
int edp_num;
|
|
unsigned int ret = ABM_FEATURE_NO_SUPPORT;
|
|
|
|
dc_get_edp_links(dc->dc, edp_links, &edp_num);
|
|
|
|
for (i = 0; i < edp_num; i++) {
|
|
if (panel_inst == i)
|
|
break;
|
|
}
|
|
|
|
if (i < edp_num) {
|
|
ret = ABM_LCD_SUPPORT;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void dmub_abm_init_ex(struct abm *abm, uint32_t backlight)
|
|
{
|
|
dmub_abm_init(abm, backlight);
|
|
}
|
|
|
|
static unsigned int dmub_abm_get_current_backlight_ex(struct abm *abm)
|
|
{
|
|
return dmub_abm_get_current_backlight(abm);
|
|
}
|
|
|
|
static unsigned int dmub_abm_get_target_backlight_ex(struct abm *abm)
|
|
{
|
|
return dmub_abm_get_target_backlight(abm);
|
|
}
|
|
|
|
static bool dmub_abm_set_level_ex(struct abm *abm, uint32_t level)
|
|
{
|
|
bool ret = false;
|
|
unsigned int feature_support, i;
|
|
uint8_t panel_mask0 = 0;
|
|
|
|
for (i = 0; i < MAX_NUM_EDP; i++) {
|
|
feature_support = abm_feature_support(abm, i);
|
|
|
|
if (feature_support == ABM_LCD_SUPPORT)
|
|
panel_mask0 |= (0x01 << i);
|
|
}
|
|
|
|
if (panel_mask0)
|
|
ret = dmub_abm_set_level(abm, level, panel_mask0);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static bool dmub_abm_init_config_ex(struct abm *abm,
|
|
const char *src,
|
|
unsigned int bytes,
|
|
unsigned int inst)
|
|
{
|
|
unsigned int feature_support;
|
|
|
|
feature_support = abm_feature_support(abm, inst);
|
|
|
|
if (feature_support == ABM_LCD_SUPPORT)
|
|
dmub_abm_init_config(abm, src, bytes, inst);
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool dmub_abm_set_pause_ex(struct abm *abm, bool pause, unsigned int panel_inst, unsigned int stream_inst)
|
|
{
|
|
bool ret = false;
|
|
unsigned int feature_support;
|
|
|
|
feature_support = abm_feature_support(abm, panel_inst);
|
|
|
|
if (feature_support == ABM_LCD_SUPPORT)
|
|
ret = dmub_abm_set_pause(abm, pause, panel_inst, stream_inst);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static bool dmub_abm_set_pipe_ex(struct abm *abm, uint32_t otg_inst, uint32_t option, uint32_t panel_inst)
|
|
{
|
|
bool ret = false;
|
|
unsigned int feature_support;
|
|
|
|
feature_support = abm_feature_support(abm, panel_inst);
|
|
|
|
if (feature_support == ABM_LCD_SUPPORT)
|
|
ret = dmub_abm_set_pipe(abm, otg_inst, option, panel_inst);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static bool dmub_abm_set_backlight_level_pwm_ex(struct abm *abm,
|
|
unsigned int backlight_pwm_u16_16,
|
|
unsigned int frame_ramp,
|
|
unsigned int controller_id,
|
|
unsigned int panel_inst)
|
|
{
|
|
bool ret = false;
|
|
unsigned int feature_support;
|
|
|
|
feature_support = abm_feature_support(abm, panel_inst);
|
|
|
|
if (feature_support == ABM_LCD_SUPPORT)
|
|
ret = dmub_abm_set_backlight_level(abm, backlight_pwm_u16_16, frame_ramp, panel_inst);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static const struct abm_funcs abm_funcs = {
|
|
.abm_init = dmub_abm_init_ex,
|
|
.set_abm_level = dmub_abm_set_level_ex,
|
|
.get_current_backlight = dmub_abm_get_current_backlight_ex,
|
|
.get_target_backlight = dmub_abm_get_target_backlight_ex,
|
|
.init_abm_config = dmub_abm_init_config_ex,
|
|
.set_abm_pause = dmub_abm_set_pause_ex,
|
|
.set_pipe_ex = dmub_abm_set_pipe_ex,
|
|
.set_backlight_level_pwm = dmub_abm_set_backlight_level_pwm_ex,
|
|
};
|
|
|
|
static void dmub_abm_construct(
|
|
struct dce_abm *abm_dce,
|
|
struct dc_context *ctx,
|
|
const struct dce_abm_registers *regs,
|
|
const struct dce_abm_shift *abm_shift,
|
|
const struct dce_abm_mask *abm_mask)
|
|
{
|
|
struct abm *base = &abm_dce->base;
|
|
|
|
base->ctx = ctx;
|
|
base->funcs = &abm_funcs;
|
|
base->dmcu_is_running = false;
|
|
|
|
abm_dce->regs = regs;
|
|
abm_dce->abm_shift = abm_shift;
|
|
abm_dce->abm_mask = abm_mask;
|
|
}
|
|
|
|
struct abm *dmub_abm_create(
|
|
struct dc_context *ctx,
|
|
const struct dce_abm_registers *regs,
|
|
const struct dce_abm_shift *abm_shift,
|
|
const struct dce_abm_mask *abm_mask)
|
|
{
|
|
if (ctx->dc->caps.dmcub_support) {
|
|
struct dce_abm *abm_dce = kzalloc(sizeof(*abm_dce), GFP_KERNEL);
|
|
|
|
if (abm_dce == NULL) {
|
|
BREAK_TO_DEBUGGER();
|
|
return NULL;
|
|
}
|
|
|
|
dmub_abm_construct(abm_dce, ctx, regs, abm_shift, abm_mask);
|
|
|
|
return &abm_dce->base;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void dmub_abm_destroy(struct abm **abm)
|
|
{
|
|
struct dce_abm *abm_dce = TO_DMUB_ABM(*abm);
|
|
|
|
kfree(abm_dce);
|
|
*abm = NULL;
|
|
}
|