[PATCH 6/8] drm/amd/display: Set DC options from modifiers.

Daniel Vetter daniel at ffwll.ch
Mon Aug 10 12:28:20 UTC 2020


On Wed, Aug 05, 2020 at 09:32:10AM +0200, daniel at ffwll.ch wrote:
> On Tue, Aug 04, 2020 at 11:31:17PM +0200, Bas Nieuwenhuizen wrote:
> > This sets the DC tiling options from the modifier, if modifiers
> > are used for the FB. This patch by itself does not expose the
> > support yet though.
> > 
> > There is not much validation yet to limit the scope of this
> > patch, but the current validation is at the same level as
> > the BO metadata path.
> > 
> > Signed-off-by: Bas Nieuwenhuizen <bas at basnieuwenhuizen.nl>
> > ---
> >  .../gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c | 109 +++++++++++++++++-
> >  1 file changed, 103 insertions(+), 6 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 6ef7f2f8acab..ac913b8f10ef 100644
> > --- a/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c
> > +++ b/drivers/gpu/drm/amd/display/amdgpu_dm/amdgpu_dm.c
> > @@ -3754,6 +3754,93 @@ fill_gfx9_plane_attributes_from_flags(struct amdgpu_device *adev,
> >  	return 0;
> >  }
> >  
> > +static bool
> > +modifier_has_dcc(uint64_t modifier)
> > +{
> > +	return IS_AMD_FMT_MOD(modifier) && AMD_FMT_MOD_GET(DCC, modifier);
> > +}
> > +
> > +static unsigned
> > +modifier_gfx9_swizzle_mode(uint64_t modifier)
> > +{
> > +	if (modifier == DRM_FORMAT_MOD_LINEAR)
> > +		return 0;
> > +
> > +	return AMD_FMT_MOD_GET(TILE, modifier);
> > +}
> > +
> > +static void
> > +fill_gfx9_tiling_info_from_modifier(const struct amdgpu_device *adev,
> > +				  union dc_tiling_info *tiling_info,
> > +				  uint64_t modifier)
> > +{
> > +	unsigned int mod_bank_xor_bits = AMD_FMT_MOD_GET(BANK_XOR_BITS, modifier);
> > +	unsigned int mod_pipe_xor_bits = AMD_FMT_MOD_GET(PIPE_XOR_BITS, modifier);
> > +	unsigned int pkrs_log2 = AMD_FMT_MOD_GET(PACKERS, modifier);
> > +	unsigned int pipes_log2 = min(4u, mod_pipe_xor_bits);
> > +
> > +	fill_gfx9_tiling_info_from_device(adev, tiling_info);
> > +
> > +	if (!IS_AMD_FMT_MOD(modifier))
> > +		return;
> > +
> > +	tiling_info->gfx9.num_pipes = 1u << pipes_log2;
> > +	tiling_info->gfx9.num_shader_engines = 1u << (mod_pipe_xor_bits - pipes_log2);
> > +
> > +	if (adev->family >= AMDGPU_FAMILY_NV) {
> > +		tiling_info->gfx9.num_pkrs = 1u << pkrs_log2;
> > +	} else {
> > +		tiling_info->gfx9.num_banks = 1u << mod_bank_xor_bits;
> > +
> > +		/* for DCC we know it isn't rb aligned, so rb_per_se doesn't matter. */
> > +	}
> > +}
> > +
> > +static void
> > +block_alignment(unsigned int blocksize_log2, unsigned int *width, unsigned int *height)
> > +{
> > +	unsigned int height_log2 = blocksize_log2 / 2;
> > +	unsigned int width_log2 = blocksize_log2 - height_log2;
> > +
> > +	*width = 1u << width_log2;
> > +	*height = 1u << height_log2;
> > +}
> > +
> > +static int
> > +fill_gfx9_plane_attributes_from_modifiers(struct amdgpu_device *adev,
> > +				      const struct amdgpu_framebuffer *afb,
> > +				      const enum surface_pixel_format format,
> > +				      const enum dc_rotation_angle rotation,
> > +				      const struct plane_size *plane_size,
> > +				      union dc_tiling_info *tiling_info,
> > +				      struct dc_plane_dcc_param *dcc,
> > +				      struct dc_plane_address *address,
> > +				      const bool force_disable_dcc)
> > +{
> > +	const uint64_t modifier = afb->base.modifier;
> > +	int ret;
> > +
> > +	fill_gfx9_tiling_info_from_modifier(adev, tiling_info, modifier);
> > +	tiling_info->gfx9.swizzle = modifier_gfx9_swizzle_mode(modifier);
> > +
> > +	if (modifier_has_dcc(modifier) && !force_disable_dcc) {
> > +		uint64_t dcc_address = afb->address + afb->base.offsets[1];
> > +
> > +		dcc->enable = 1;
> > +		dcc->meta_pitch = afb->base.pitches[1];
> > +		dcc->independent_64b_blks = AMD_FMT_MOD_GET(DCC_INDEPENDENT_64B, modifier);
> > +
> > +		address->grph.meta_addr.low_part = lower_32_bits(dcc_address);
> > +		address->grph.meta_addr.high_part = upper_32_bits(dcc_address);
> > +	}
> > +
> > +	ret = validate_dcc(adev, format, rotation, tiling_info, dcc, address, plane_size);
> > +	if (ret)
> > +		return ret;
> > +
> > +	return 0;
> > +}
> > +
> >  static int
> >  fill_plane_buffer_attributes(struct amdgpu_device *adev,
> >  			     const struct amdgpu_framebuffer *afb,
> > @@ -3823,12 +3910,22 @@ fill_plane_buffer_attributes(struct amdgpu_device *adev,
> >  
> >  
> >  	if (adev->family >= AMDGPU_FAMILY_AI) {
> > -		ret = fill_gfx9_plane_attributes_from_flags(adev, afb, format, rotation,
> > -							    plane_size, tiling_info, dcc,
> > -							    address, tiling_flags,
> > -							    force_disable_dcc);
> > -		if (ret)
> > -			return ret;
> > +		if (afb->base.flags & DRM_MODE_FB_MODIFIERS) {
> > +			ret = fill_gfx9_plane_attributes_from_modifiers(adev, afb, format,
> > +								      rotation, plane_size,
> > +								      tiling_info, dcc,
> > +								      address,
> > +								      force_disable_dcc);
> > +			if (ret)
> > +				return ret;
> > +		} else {
> > +			ret = fill_gfx9_plane_attributes_from_flags(adev, afb, format, rotation,
> > +								  plane_size, tiling_info, dcc,
> > +								  address, tiling_flags,
> > +								  force_disable_dcc);
> > +			if (ret)
> > +				return ret;
> 
> So what we've done in i915, but might be too cumbersome with the amdgpu
> modifiers, is to map legacy tiling information into modifiers once, at the
> beginning of addfb. So in amdgpu_display_user_framebuffer_create(). And
> once modifiers are filled in, you ofc set the DRM_MODE_FB_MODIFIERS flag
> too in the fb.
> 
> Then all the display code only works with modifiers, instead of having a
> mix and possible confusion, with breakage when people only test the legacy
> path. Which I kinda expect to happen, since amd probably runs with their
> own ddx in their CI rig only.
> 
> This also avoids a bunch of layering and locking unprettiness, since
> display code doesn't need to dig around in gem_bo side of things. On that,
> there's another amdgpu_bo_get_tiling_flags in amdgpu_dm_commit_planes
> which probably shouldn't be there, and should use computed stuff from
> plane state or fb (I changed it to a lockless version to just hack around
> locking issues, but still there).
> 
> This hopefully/eventually should allow us to entirely phase out the legacy
> magic tiling blob attached to bo (we've pulled the trigger on that for
> intel now, would have needed to extend the uapi to keep it working was a
> good excuse).

Ok just learned that amdgpu hat set/get_tiling, so I'm upgrading my idea
here to a very strong recommendation, i.e. please do this except if
there's and amd ddx which somehow wants to change tiling mode while a fb
exists, and expects this to propagate.

In i915 we even disallow the set_tiling ioctl with an error if an fb
exists, just to make sure userspace behaves. Even if userspace uses
set_tiling, this way we can at least enforce the same semantics of "client
can't pull compositor over the table with a set_tiling at the wrong time"
of modifiers.
-Daniel

> 
> Cheers, Daniel
> 
> > +		}
> >  	} else {
> >  		fill_gfx8_tiling_info_from_flags(tiling_info, tiling_flags);
> >  	}
> > -- 
> > 2.28.0
> > 
> 
> -- 
> Daniel Vetter
> Software Engineer, Intel Corporation
> http://blog.ffwll.ch

-- 
Daniel Vetter
Software Engineer, Intel Corporation
http://blog.ffwll.ch


More information about the amd-gfx mailing list