Received: by 2002:a05:6a10:a0d1:0:0:0:0 with SMTP id j17csp605209pxa; Tue, 4 Aug 2020 13:16:32 -0700 (PDT) X-Google-Smtp-Source: ABdhPJzAZudYRi/SHYMowKbNFQ7TYB9XghU759xs88BJYti9AWlWLw1cVw4nMRjxRQ+lNV0TScOg X-Received: by 2002:a05:6402:1d32:: with SMTP id dh18mr22023968edb.14.1596572192043; Tue, 04 Aug 2020 13:16:32 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1596572192; cv=none; d=google.com; s=arc-20160816; b=dJnQKCBgyhUQJh6XB+FqQLC+sRqz5+ftbPX7/zqe58JAPPHFykwZmX5jCuzFcqspwt /MjNL6TQTBqmwymcUv57+FqG2xLOKcOCnJze1lbIhEkJ10si6kmlj2LqIboUn6IfCykD Ds/4R5mJtcMtqIkEdJFjBsbeG9jCt2RZl5JstGBoBEX24i2MSdKNRa0R/YhYAW/9kaqW vAlHlkc+HAD62PKe4A61MwpXVvuGQ8l2ysf0LWBiX02i/nQdLS/IlNcoLT8htvv8OANi khsfun/qAwKbATVnDathMIX1b0fzeHSRMRm2SNFyTUDR53Y5EoIn/aYVhZI8c8k/Zy2H u8BA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :message-id:date:subject:cc:to:from; bh=uWs57Va/p4SBIv6cr8jGvxmO6bBAjpVcL/tHkdd/sGs=; b=s1XnVkfK3NkpgpNlaf39/CbMz9CJ8bRvyuNZyiBJAkd7ZcJMmCc2qyUKDRXBzIE/ES Fp49kQrmd4tUCEf9ONTDTRjSufVhHtbWCxemD/f7qLNSuPsiN23UmeWq+uB00T4hAm37 Ek5KsuNLfsYbfEZIwhJkqfQCCO51ANJsfj4MvB5mPm7sGJgoD/rRmHpQz5CRbEVnfaPj lXaF0UKrd6vDCjZjWws6azoG6TC5Zq45ZYL90VmENYtE2CRDFwOg7bZKEnnEQccmfNiw FahrwWsxqa54KV0UZtgFXIl+slztM0HZu7JesDidu95HBv05ZiDsS4+DLTUFJA971cG/ n45A== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=ispras.ru Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id v6si14185798edi.91.2020.08.04.13.16.07; Tue, 04 Aug 2020 13:16:32 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=ispras.ru Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726631AbgHDUNc (ORCPT + 99 others); Tue, 4 Aug 2020 16:13:32 -0400 Received: from mail.ispras.ru ([83.149.199.84]:51862 "EHLO mail.ispras.ru" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1725981AbgHDUNc (ORCPT ); Tue, 4 Aug 2020 16:13:32 -0400 Received: from localhost.localdomain (unknown [46.188.10.168]) by mail.ispras.ru (Postfix) with ESMTPSA id ADE5240A2071; Tue, 4 Aug 2020 20:13:28 +0000 (UTC) From: Alexander Monakov To: amd-gfx@lists.freedesktop.org Cc: linux-kernel@vger.kernel.org, Alexander Monakov , Alex Deucher , Nicholas Kazlauskas Subject: [PATCH v2] drm/amd/display: use correct scale for actual_brightness Date: Tue, 4 Aug 2020 23:13:13 +0300 Message-Id: <20200804201313.6464-1-amonakov@ispras.ru> X-Mailer: git-send-email 2.26.2 MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Documentation for sysfs backlight level interface requires that values in both 'brightness' and 'actual_brightness' files are interpreted to be in range from 0 to the value given in the 'max_brightness' file. With amdgpu, max_brightness gives 255, and values written by the user into 'brightness' are internally rescaled to a wider range. However, reading from 'actual_brightness' gives the raw register value without inverse rescaling. This causes issues for various userspace tools such as PowerTop and systemd that expect the value to be in the correct range. Introduce a helper to retrieve internal backlight range. Use it to reimplement 'convert_brightness' as 'convert_brightness_from_user' and introduce 'convert_brightness_to_user'. Bug: https://bugzilla.kernel.org/show_bug.cgi?id=203905 Bug: https://gitlab.freedesktop.org/drm/amd/-/issues/1242 Cc: Alex Deucher Cc: Nicholas Kazlauskas Signed-off-by: Alexander Monakov --- v2: split convert_brightness to &_from_user and &_to_user (Nicholas) .../gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c | 81 +++++++++---------- 1 file changed, 40 insertions(+), 41 deletions(-) diff --git a/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c b/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c index 710edc70e37e..b60a763f3f95 100644 --- a/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c +++ b/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c @@ -2881,51 +2881,50 @@ static int set_backlight_via_aux(struct dc_link *link, uint32_t brightness) return rc ? 0 : 1; } -static u32 convert_brightness(const struct amdgpu_dm_backlight_caps *caps, - const uint32_t user_brightness) +static int get_brightness_range(const struct amdgpu_dm_backlight_caps *caps, + unsigned *min, unsigned *max) { - u32 min, max, conversion_pace; - u32 brightness = user_brightness; - if (!caps) - goto out; + return 0; - if (!caps->aux_support) { - max = caps->max_input_signal; - min = caps->min_input_signal; - /* - * The brightness input is in the range 0-255 - * It needs to be rescaled to be between the - * requested min and max input signal - * It also needs to be scaled up by 0x101 to - * match the DC interface which has a range of - * 0 to 0xffff - */ - conversion_pace = 0x101; - brightness = - user_brightness - * conversion_pace - * (max - min) - / AMDGPU_MAX_BL_LEVEL - + min * conversion_pace; + if (caps->aux_support) { + // Firmware limits are in nits, DC API wants millinits. + *max = 1000 * caps->aux_max_input_signal; + *min = 1000 * caps->aux_min_input_signal; } else { - /* TODO - * We are doing a linear interpolation here, which is OK but - * does not provide the optimal result. We probably want - * something close to the Perceptual Quantizer (PQ) curve. - */ - max = caps->aux_max_input_signal; - min = caps->aux_min_input_signal; - - brightness = (AMDGPU_MAX_BL_LEVEL - user_brightness) * min - + user_brightness * max; - // Multiple the value by 1000 since we use millinits - brightness *= 1000; - brightness = DIV_ROUND_CLOSEST(brightness, AMDGPU_MAX_BL_LEVEL); + // Firmware limits are 8-bit, PWM control is 16-bit. + *max = 0x101 * caps->max_input_signal; + *min = 0x101 * caps->min_input_signal; } + return 1; +} -out: - return brightness; +static u32 convert_brightness_from_user(const struct amdgpu_dm_backlight_caps *caps, + uint32_t brightness) +{ + unsigned min, max; + + if (!get_brightness_range(caps, &min, &max)) + return brightness; + + // Rescale 0..255 to min..max + return min + DIV_ROUND_CLOSEST((max - min) * brightness, + AMDGPU_MAX_BL_LEVEL); +} + +static u32 convert_brightness_to_user(const struct amdgpu_dm_backlight_caps *caps, + uint32_t brightness) +{ + unsigned min, max; + + if (!get_brightness_range(caps, &min, &max)) + return brightness; + + if (brightness < min) + return 0; + // Rescale min..max to 0..255 + return DIV_ROUND_CLOSEST(AMDGPU_MAX_BL_LEVEL * (brightness - min), + max - min); } static int amdgpu_dm_backlight_update_status(struct backlight_device *bd) @@ -2941,7 +2940,7 @@ static int amdgpu_dm_backlight_update_status(struct backlight_device *bd) link = (struct dc_link *)dm->backlight_link; - brightness = convert_brightness(&caps, bd->props.brightness); + brightness = convert_brightness_from_user(&caps, bd->props.brightness); // Change brightness based on AUX property if (caps.aux_support) return set_backlight_via_aux(link, brightness); @@ -2958,7 +2957,7 @@ static int amdgpu_dm_backlight_get_brightness(struct backlight_device *bd) if (ret == DC_ERROR_UNEXPECTED) return bd->props.brightness; - return ret; + return convert_brightness_to_user(&dm->backlight_caps, ret); } static const struct backlight_ops amdgpu_dm_backlight_ops = { base-commit: bcf876870b95592b52519ed4aafcf9d95999bc9c -- 2.26.2