sam_tc_clockselect() reworked to calculate frequency error using smallest possible divisor minimizing the frequency error rather than largest possible divisor which maximized the error.

This commit is contained in:
Piotr Mienkowski 2016-08-17 09:51:54 -06:00 committed by Gregory Nutt
parent 17e5da96ea
commit 5d5851a5cd
6 changed files with 104 additions and 202 deletions

View File

@ -908,9 +908,9 @@ Build Targets and Options
distclean
Does 'clean' then also removes all configuration and context files.
This essentially restores the directory structure to its original,
unconfigured stated.
Does 'clean' then also removes all configuration, dependency, and
other context files. This essentially restores the directory structure
to its original, unconfigured stated.
Application housekeeping targets. The APPDIR variable refers to the user
application directory. A sample apps/ directory is included with NuttX,

View File

@ -373,7 +373,8 @@ static int dac_timer_init(struct sam_dac_s *priv, uint32_t freq_required,
int channel)
{
uint32_t mode;
uint32_t regval;
uint32_t tcclks;
uint32_t div;
uint32_t freq_actual;
ainfo("required frequency=%ld [Hz], channel=%d\n",
@ -381,15 +382,11 @@ static int dac_timer_init(struct sam_dac_s *priv, uint32_t freq_required,
DEBUGASSERT(priv && (freq_required > 0) && (channel >= 0 && channel <= 2));
/* Set the timer/counter waveform mode the the clock input. Use smallest
* MCK divisor of 8 to have highest clock resolution thus smallest frequency
* error. With 32 bit counter the lowest possible frequency of 1 Hz is easily
* supported.
*/
/* Calculate the best possible clock source and clock divisor value */
/* TODO Add support for TC_CMR_TCCLKS_PCK6 to reduce frequency error */
freq_actual = sam_tc_clockselect(freq_required, &tcclks, &div);
mode = (TC_CMR_TCCLKS_MCK8 | /* Use MCK/8 clock signal */
mode = (tcclks | /* Use MCK/8 clock signal */
TC_CMR_WAVSEL_UPRC | /* UP mode w/ trigger on RC Compare */
TC_CMR_WAVE | /* Wave mode */
TC_CMR_ACPA_CLEAR | /* RA Compare Effect on TIOA: Clear */
@ -404,21 +401,17 @@ static int dac_timer_init(struct sam_dac_s *priv, uint32_t freq_required,
return -EINVAL;
}
/* Calculate the actual counter value from this divider and the tc input
* frequency.
*/
/* Set up clock divisor */
regval = BOARD_MCK_FREQUENCY / 8 / freq_required;
DEBUGASSERT(regval > 0); /* Will check for integer underflow */
DEBUGASSERT(div >= 2); /* Minimum divider required by implementation */
/* Set up TC_RA and TC_RC. The frequency is determined by RA and RC:
* TIOA is cleared on RA match; TIOA is set on RC match.
*/
sam_tc_setregister(priv->tc, TC_REGA, regval >> 1);
sam_tc_setregister(priv->tc, TC_REGC, regval);
sam_tc_setregister(priv->tc, TC_REGA, div >> 1);
sam_tc_setregister(priv->tc, TC_REGC, div);
freq_actual = BOARD_MCK_FREQUENCY / 8 / regval;
ainfo("configured frequency=%ld [Hz]\n", (long)freq_actual);
/* And start the timer */

View File

@ -119,29 +119,24 @@ static void sam_freerun_handler(TC_HANDLE tch, void *arg, uint32_t sr)
int sam_freerun_initialize(struct sam_freerun_s *freerun, int chan,
uint16_t resolution)
{
uint32_t frequency;
uint32_t actual;
uint32_t freq_required;
uint32_t freq_actual;
uint32_t div;
uint32_t cmr;
int ret;
tmrinfo("chan=%d resolution=%d usec\n", chan, resolution);
DEBUGASSERT(freerun && resolution > 0);
/* Get the TC frequency the corresponds to the requested resolution */
frequency = USEC_PER_SEC / (uint32_t)resolution;
freq_required = USEC_PER_SEC / (uint32_t)resolution;
/* The pre-calculate values to use when we start the timer */
ret = sam_tc_clockselect(frequency, &cmr, &actual);
if (ret < 0)
{
tmrerr("ERROR: sam_tc_clockselect failed: %d\n", ret);
return ret;
}
freq_actual = sam_tc_clockselect(freq_required, &cmr, &div);
tmrinfo("frequency=%lu, actual=%lu, cmr=%08lx\n",
(unsigned long)frequency, (unsigned long)actual,
tmrinfo("freq required=%lu, freq actual=%lu, TC_CMR.TCCLKS=%08lx\n",
(unsigned long)freq_required, (unsigned long)freq_actual,
(unsigned long)cmr);
/* Allocate the timer/counter and select its mode of operation

View File

@ -147,29 +147,24 @@ static void sam_oneshot_handler(TC_HANDLE tch, void *arg, uint32_t sr)
int sam_oneshot_initialize(struct sam_oneshot_s *oneshot, int chan,
uint16_t resolution)
{
uint32_t frequency;
uint32_t actual;
uint32_t freq_required;
uint32_t freq_actual;
uint32_t div;
uint32_t cmr;
int ret;
tmrinfo("chan=%d resolution=%d usec\n", chan, resolution);
DEBUGASSERT(oneshot && resolution > 0);
/* Get the TC frequency the corresponds to the requested resolution */
frequency = USEC_PER_SEC / (uint32_t)resolution;
freq_required = USEC_PER_SEC / (uint32_t)resolution;
/* The pre-calculate values to use when we start the timer */
ret = sam_tc_clockselect(frequency, &cmr, &actual);
if (ret < 0)
{
tmrerr("ERROR: sam_tc_clockselect failed: %d\n", ret);
return ret;
}
freq_actual = sam_tc_clockselect(freq_required, &cmr, &div);
tmrinfo("frequency=%lu, actual=%lu, cmr=%08lx\n",
(unsigned long)frequency, (unsigned long)actual,
tmrinfo("freq required=%lu, freq actual=%lu, TC_CMR.TCCLKS=%08lx\n",
(unsigned long)freq_required, (unsigned long)freq_actual,
(unsigned long)cmr);
/* Allocate the timer/counter and select its mode of operation

View File

@ -213,8 +213,8 @@ static int sam_tc11_interrupt(int irq, void *context);
static uint32_t sam_tc_mckfreq_lookup(uint32_t ftcin, int ndx);
static inline uint32_t sam_tc_tcclks_lookup(int ndx);
static int sam_tc_mcksrc(uint32_t frequency, uint32_t *tcclks,
uint32_t *actual);
static uint32_t sam_tc_freq_err_abs(uint32_t freq_required,
uint32_t freq_input, uint32_t *div);
static inline struct sam_chan_s *sam_tc_initialize(int channel);
/****************************************************************************
@ -1019,83 +1019,51 @@ static inline uint32_t sam_tc_tcclks_lookup(int ndx)
}
/****************************************************************************
* Name: sam_tc_mcksrc
* Name: sam_tc_freq_err_abs
*
* Description:
* Finds the best MCK divisor given the timer frequency and MCK. The
* result is guaranteed to satisfy the following equation:
*
* (Ftcin / (div * 65536)) <= freq <= (Ftcin / div)
*
* where:
* freq - the desired frequency
* Ftcin - The timer/counter input frequency
* div - With DIV being the highest possible value.
* Calculate best possible frequency error given input frequency and
* required frequency knowing that input frequency can be divided by
* integer divisor. The divisor for which the given error was calculated
* is also returned.
*
* Input Parameters:
* frequency Desired timer frequency.
* tcclks TCCLKS field value for divisor.
* actual The actual freqency of the MCK
* freq_required Desired timer frequency
* freq_input TC module input frequency
* div Pointer to the divisor for which the error was
* calculated
*
* Returned Value:
* Zero (OK) if a proper divisor has been found, otherwise a negated errno
* value indicating the nature of the failure.
* Absolute value of the smallest possible frequency error
*
****************************************************************************/
static int sam_tc_mcksrc(uint32_t frequency, uint32_t *tcclks,
uint32_t *actual)
static uint32_t sam_tc_freq_err_abs(uint32_t freq_required, uint32_t freq_input,
uint32_t *div)
{
uint32_t fselect;
uint32_t fnext;
int ndx = 0;
uint32_t freq_actual;
uint32_t freq_error;
tmrinfo("frequency=%d\n", frequency);
DEBUGASSERT(freq_input >= freq_required);
DEBUGASSERT(UINT32_MAX - freq_required/2 > freq_input);
/* Satisfy lower bound. That is, the value of the divider such that:
*
* frequency >= (tc_input_frequency * 65536) / divider.
/* Integer division will truncate result toward zero, make sure the result
* is rounded instead.
*/
for (; ndx < TC_NDIVIDERS; ndx++)
*div = (freq_input + freq_required/2) / freq_required;
freq_actual = freq_input / *div;
if (freq_required >= freq_actual)
{
fselect = sam_tc_mckfreq_lookup(BOARD_MCK_FREQUENCY, ndx);
if (frequency >= (fselect >> 16))
{
break;
freq_error = freq_required - freq_actual;
}
else
{
freq_error = freq_actual - freq_required;
}
if (ndx >= TC_NDIVIDERS)
{
/* If no divisor can be found, return -ERANGE */
tmrerr("ERROR: Lower bound search failed\n");
return -ERANGE;
}
/* Try to maximize DIV while still satisfying upper bound. That the
* value of the divider such that:
*
* frequency < tc_input_frequency / divider.
*/
for (; ndx < TC_NDIVIDERS; ndx++)
{
fnext = sam_tc_mckfreq_lookup(BOARD_MCK_FREQUENCY, ndx + 1);
if (frequency > fnext)
{
break;
}
fselect = fnext;
}
/* Return the actual frequency and the TCCLKS selection */
*actual = fselect;
*tcclks = sam_tc_tcclks_lookup(ndx);
return OK;
return freq_error;
}
/****************************************************************************
@ -1644,120 +1612,80 @@ uint32_t sam_tc_divfreq(TC_HANDLE handle)
* Name: sam_tc_clockselect
*
* Description:
* Finds the best MCK divisor given the timer frequency and MCK. The
* result is guaranteed to satisfy the following equation:
*
* (Ftcin / (div * 65536)) <= freq <= (Ftcin / div)
*
* where:
* freq - the desired frequency
* Ftcin - The timer/counter input frequency
* div - With DIV being the highest possible value.
* Finds the best clock source and clock divisor to configure required
* frequency.
*
* Input Parameters:
* frequency Desired timer frequency.
* tcclks TCCLKS field value for divisor.
* actual The actual freqency of the MCK
* frequency Desired timer frequency
* tcclks TC_CMRx.TCCLKS bit field (clock selection) value
* div The divisor value to be configured for the TC
*
* Returned Value:
* Zero (OK) if a proper divisor has been found, otherwise a negated errno
* value indicating the nature of the failure.
* Rhe actual frequency which will be configured with calculated
* parameters
*
****************************************************************************/
int sam_tc_clockselect(uint32_t frequency, uint32_t *tcclks,
uint32_t *actual)
uint32_t sam_tc_clockselect(uint32_t frequency, uint32_t *tcclks,
uint32_t *div)
{
uint32_t mck_actual;
uint32_t mck_tcclks;
uint32_t mck_error;
int ret;
uint32_t mck8_freq;
uint32_t mck8_error;
uint32_t tcclks_select;
uint32_t div_select;
uint32_t freq_actual;
/* Try to satisfy the requested frequency with the MCK or slow clock */
/* Calculate frequency error for MCK clock. Use smallest possible MCK
* divisor of 8 to have highest clock resolution and thus smallest
* frequency error. With 32 bit counter the lowest possible frequency
* of 1 Hz is easily supported.
*/
ret = sam_tc_mcksrc(frequency, &mck_tcclks, &mck_actual);
if (ret < 0)
{
mck_error = UINT32_MAX;
}
else
{
/* Get the absolute value of the frequency error */
if (mck_actual > frequency)
{
mck_error = mck_actual - frequency;
}
else
{
mck_error = frequency - mck_actual;
}
}
mck8_freq = BOARD_MCK_FREQUENCY/8;
mck8_error = sam_tc_freq_err_abs(frequency, mck8_freq, &div_select);
tcclks_select = TC_CMR_TCCLKS_MCK8;
freq_actual = mck8_freq / div_select;
/* See if we do better with PCK6 */
if (sam_pck_isenabled(PCK6))
{
uint32_t pck6_actual;
uint32_t pck6_freq;
uint32_t pck6_error;
uint32_t pck6_div;
/* Get the absolute value of the frequency error */
pck6_actual = sam_pck_frequency(PCK6);
if (pck6_actual > frequency)
{
pck6_error = pck6_actual - frequency;
}
else
{
pck6_error = frequency - pck6_actual;
}
pck6_freq = sam_pck_frequency(PCK6);
pck6_error = sam_tc_freq_err_abs(frequency, pck6_freq, &pck6_div);
/* Return the PCK6 selection if the error is smaller */
if (pck6_error < mck_error)
if (pck6_error < mck8_error)
{
/* Return the PCK selection */
if (actual)
{
tmrinfo("return actual=%lu\n", (unsigned long)fselect);
*actual = pck6_actual;
tcclks_select = TC_CMR_TCCLKS_PCK6;
div_select = pck6_div;
freq_actual = pck6_freq / pck6_div;
}
}
/* Return the TCCLKS selection */
if (tcclks)
{
tmrinfo("return tcclks=%08lx\n", (unsigned long)TC_CMR_TCCLKS_PCK6);
*tcclks = TC_CMR_TCCLKS_PCK6;
tmrinfo("return tcclks=%08lx\n", (unsigned long)tcclks_select);
*tcclks = tcclks_select;
}
/* Return success */
/* Return the divider value */
return OK;
}
}
/* Return the MCK/slow clock selection */
if (actual)
if (div)
{
tmrinfo("return actual=%lu\n", (unsigned long)mck_actual);
*actual = mck_actual;
tmrinfo("return div=%lu\n", (unsigned long)div_select);
*div = div_select;
}
/* Return the TCCLKS selection */
if (tcclks)
{
tmrinfo("return tcclks=%08lx\n", (unsigned long)mck_tcclks);
*tcclks = mck_tcclks;
}
/* Return success */
return ret;
return freq_actual;
}
#endif /* CONFIG_SAMV7_TC0 || CONFIG_SAMV7_TC1 || CONFIG_SAMV7_TC2 || CONFIG_SAMV7_TC3 */

View File

@ -319,29 +319,20 @@ uint32_t sam_tc_divfreq(TC_HANDLE handle);
* Name: sam_tc_clockselect
*
* Description:
* Finds the best MCK divisor given the timer frequency and MCK. The
* result is guaranteed to satisfy the following equation:
*
* (Ftcin / (div * 65536)) <= freq <= (Ftcin / div)
*
* where:
* freq - the desired frequency
* Ftcin - The timer/counter input frequency
* div - With DIV being the highest possible value.
* Finds the best clock source and clock divisor to configure required
* frequency.
*
* Input Parameters:
* frequency Desired timer frequency.
* tcclks TCCLKS field value for divisor.
* actual The actual freqency of the MCK
* frequency desired timer frequency
* tcclks TC_CMRx.TCCLKS bit field (clock selection) value
* div the divisor value to be configured for the TC
*
* Returned Value:
* Zero (OK) if a proper divisor has been found, otherwise a negated errno
* value indicating the nature of the failure.
* the actual frequency which will be configured with calculated parameters
*
****************************************************************************/
int sam_tc_clockselect(uint32_t frequency, uint32_t *tcclks,
uint32_t *actual);
uint32_t sam_tc_clockselect(uint32_t frequency, uint32_t *tcclks, uint32_t *div);
#undef EXTERN
#ifdef __cplusplus