162306a36Sopenharmony_ci/* 262306a36Sopenharmony_ci * Copyright 2008 Advanced Micro Devices, Inc. 362306a36Sopenharmony_ci * Copyright 2008 Red Hat Inc. 462306a36Sopenharmony_ci * Copyright 2009 Christian König. 562306a36Sopenharmony_ci * 662306a36Sopenharmony_ci * Permission is hereby granted, free of charge, to any person obtaining a 762306a36Sopenharmony_ci * copy of this software and associated documentation files (the "Software"), 862306a36Sopenharmony_ci * to deal in the Software without restriction, including without limitation 962306a36Sopenharmony_ci * the rights to use, copy, modify, merge, publish, distribute, sublicense, 1062306a36Sopenharmony_ci * and/or sell copies of the Software, and to permit persons to whom the 1162306a36Sopenharmony_ci * Software is furnished to do so, subject to the following conditions: 1262306a36Sopenharmony_ci * 1362306a36Sopenharmony_ci * The above copyright notice and this permission notice shall be included in 1462306a36Sopenharmony_ci * all copies or substantial portions of the Software. 1562306a36Sopenharmony_ci * 1662306a36Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 1762306a36Sopenharmony_ci * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 1862306a36Sopenharmony_ci * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 1962306a36Sopenharmony_ci * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR 2062306a36Sopenharmony_ci * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 2162306a36Sopenharmony_ci * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 2262306a36Sopenharmony_ci * OTHER DEALINGS IN THE SOFTWARE. 2362306a36Sopenharmony_ci * 2462306a36Sopenharmony_ci * Authors: Christian König 2562306a36Sopenharmony_ci */ 2662306a36Sopenharmony_ci#include <linux/hdmi.h> 2762306a36Sopenharmony_ci#include <linux/gcd.h> 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci#include <drm/amdgpu_drm.h> 3062306a36Sopenharmony_ci#include "amdgpu.h" 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_cistatic const struct amdgpu_afmt_acr amdgpu_afmt_predefined_acr[] = { 3362306a36Sopenharmony_ci /* 32kHz 44.1kHz 48kHz */ 3462306a36Sopenharmony_ci /* Clock N CTS N CTS N CTS */ 3562306a36Sopenharmony_ci { 25175, 4096, 25175, 28224, 125875, 6144, 25175 }, /* 25,20/1.001 MHz */ 3662306a36Sopenharmony_ci { 25200, 4096, 25200, 6272, 28000, 6144, 25200 }, /* 25.20 MHz */ 3762306a36Sopenharmony_ci { 27000, 4096, 27000, 6272, 30000, 6144, 27000 }, /* 27.00 MHz */ 3862306a36Sopenharmony_ci { 27027, 4096, 27027, 6272, 30030, 6144, 27027 }, /* 27.00*1.001 MHz */ 3962306a36Sopenharmony_ci { 54000, 4096, 54000, 6272, 60000, 6144, 54000 }, /* 54.00 MHz */ 4062306a36Sopenharmony_ci { 54054, 4096, 54054, 6272, 60060, 6144, 54054 }, /* 54.00*1.001 MHz */ 4162306a36Sopenharmony_ci { 74176, 4096, 74176, 5733, 75335, 6144, 74176 }, /* 74.25/1.001 MHz */ 4262306a36Sopenharmony_ci { 74250, 4096, 74250, 6272, 82500, 6144, 74250 }, /* 74.25 MHz */ 4362306a36Sopenharmony_ci { 148352, 4096, 148352, 5733, 150670, 6144, 148352 }, /* 148.50/1.001 MHz */ 4462306a36Sopenharmony_ci { 148500, 4096, 148500, 6272, 165000, 6144, 148500 }, /* 148.50 MHz */ 4562306a36Sopenharmony_ci}; 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_ci/* 4962306a36Sopenharmony_ci * calculate CTS and N values if they are not found in the table 5062306a36Sopenharmony_ci */ 5162306a36Sopenharmony_cistatic void amdgpu_afmt_calc_cts(uint32_t clock, int *CTS, int *N, int freq) 5262306a36Sopenharmony_ci{ 5362306a36Sopenharmony_ci int n, cts; 5462306a36Sopenharmony_ci unsigned long div, mul; 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci /* Safe, but overly large values */ 5762306a36Sopenharmony_ci n = 128 * freq; 5862306a36Sopenharmony_ci cts = clock * 1000; 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_ci /* Smallest valid fraction */ 6162306a36Sopenharmony_ci div = gcd(n, cts); 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci n /= div; 6462306a36Sopenharmony_ci cts /= div; 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci /* 6762306a36Sopenharmony_ci * The optimal N is 128*freq/1000. Calculate the closest larger 6862306a36Sopenharmony_ci * value that doesn't truncate any bits. 6962306a36Sopenharmony_ci */ 7062306a36Sopenharmony_ci mul = ((128*freq/1000) + (n-1))/n; 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_ci n *= mul; 7362306a36Sopenharmony_ci cts *= mul; 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci /* Check that we are in spec (not always possible) */ 7662306a36Sopenharmony_ci if (n < (128*freq/1500)) 7762306a36Sopenharmony_ci pr_warn("Calculated ACR N value is too small. You may experience audio problems.\n"); 7862306a36Sopenharmony_ci if (n > (128*freq/300)) 7962306a36Sopenharmony_ci pr_warn("Calculated ACR N value is too large. You may experience audio problems.\n"); 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci *N = n; 8262306a36Sopenharmony_ci *CTS = cts; 8362306a36Sopenharmony_ci 8462306a36Sopenharmony_ci DRM_DEBUG("Calculated ACR timing N=%d CTS=%d for frequency %d\n", 8562306a36Sopenharmony_ci *N, *CTS, freq); 8662306a36Sopenharmony_ci} 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_cistruct amdgpu_afmt_acr amdgpu_afmt_acr(uint32_t clock) 8962306a36Sopenharmony_ci{ 9062306a36Sopenharmony_ci struct amdgpu_afmt_acr res; 9162306a36Sopenharmony_ci u8 i; 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_ci /* Precalculated values for common clocks */ 9462306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(amdgpu_afmt_predefined_acr); i++) { 9562306a36Sopenharmony_ci if (amdgpu_afmt_predefined_acr[i].clock == clock) 9662306a36Sopenharmony_ci return amdgpu_afmt_predefined_acr[i]; 9762306a36Sopenharmony_ci } 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci /* And odd clocks get manually calculated */ 10062306a36Sopenharmony_ci amdgpu_afmt_calc_cts(clock, &res.cts_32khz, &res.n_32khz, 32000); 10162306a36Sopenharmony_ci amdgpu_afmt_calc_cts(clock, &res.cts_44_1khz, &res.n_44_1khz, 44100); 10262306a36Sopenharmony_ci amdgpu_afmt_calc_cts(clock, &res.cts_48khz, &res.n_48khz, 48000); 10362306a36Sopenharmony_ci 10462306a36Sopenharmony_ci return res; 10562306a36Sopenharmony_ci} 106