On Wed, Aug 05, 2020 at 09:32:10AM +0200, daniel@xxxxxxxx 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@xxxxxxxxxxxxxxxxxxx> > > --- > > .../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 _______________________________________________ amd-gfx mailing list amd-gfx@xxxxxxxxxxxxxxxxxxxxx https://lists.freedesktop.org/mailman/listinfo/amd-gfx