Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: Генератор и индикация на MSP430
Форум разработчиков электроники ELECTRONIX.ru > Микроконтроллеры (MCs) > MSP430
Cruz21
Вообщем у меня одна небольшая проблема, не судите строго, я всего лишь студент...
Собрал я устройство, фото прикрепил. Называется "Генератор прямоугольных импульсов с регулируемой частотой и скважностью с выводом частоты на индикатор на микроконтроллере MSP430 (msp430g2553)"
Я разобрался как сделать сам генератор прямоугольных импульсов, регуляторы настроил, разобрался с индикацией. Но проблема вот в чем:
Когда я меняю сопротивление на переменном резисторе, я меняю либо частоту, либо скважность. За изменение значений отвечает АЦП (ADC10) которые свои значения заносит в свой буфер ADC10MEM. Моя задача получить эти значения и перевести его в 10 систему счисления, но это не трудно. Вся сложность состоит как конвертировать полученные значения с ADC10MEM в Гц? И как все это сделать чтоб он выводил на индикатор?
Листинг программы:
Код
#include <msp430g2553.h>

#define A0 BIT0
#define A1 BIT1
#define IND1 BIT5 //управление разрядами индикатора
#define IND2 BIT4
#define IND3 BIT3
#define IND4 BIT2

unsigned char NUM;//отвечает какой разряд горит
unsigned char BCD;//значение частоты
unsigned char BC_H1,BC_L1,BC_H2,BC_L2; //разделение по BCD по знаково
int t2ms=0;


unsigned char digits[]=
{
       0xFC, //0
       0xA0, //1
       0xDA, //2
       0xF2, //3
       0xA6, //4
       0x76, //5
       0x7E, //6
       0xE0, //7
       0xFE, //8
       0xF6, //9
       0x00, //без знака
  };

void main(void)
{
  WDTCTL = WDTPW + WDTHOLD;
  IE1 |=WDTIE;
  BCSCTL1 = CALBC1_1MHZ;  // Устанавливаем частоту DCO на калиброванные 1 MHz.
  DCOCTL = CALDCO_1MHZ;
  TACTL = TASSEL_2 + ID_3 + MC_1;
  TACCTL1 = OUTMOD_7;
  ADC10CTL1 = INCH_1 + CONSEQ_3;
  ADC10CTL0 = ADC10SHT_2 + MSC + ADC10ON + SREF_0 + ADC10IE;
  ADC10AE0 |= A0 + A1;
  P1DIR |=BIT6;
  P1SEL |=BIT6;
  __enable_interrupt();
  while(1)
  {
    ADC10CTL0 &=~ENC;
    while(ADC10CTL1 & BUSY);
    ADC10SA = 0x200;
    ADC10CTL0 |= ENC + ADC10SC;
    TACCR0 = ADC10MEM;
    TACCR1 = ADC10MEM;
  }
}

void idikacia()
{
  
  P2SEL = 0x00;
  P1DIR |= 0x3C;      // P1 output
  P2DIR |= 0xFF;      // P2 output
  
TA0CCR2 = 32000;
TA0CCTL2 = CCIE;//
TACTL = TASSEL_2 + MC_1; // SMCLK, upmode
__enable_interrupt();

_BIS_SR(LPM0_bits + GIE); // Enter LPM0 w/ interrupt
}

#pragma vector=TIMER0_A0_VECTOR
__interrupt void Timer0_A0 (void)
{
   idikacia();
   P1OUT = 0xFF;
  
switch (NUM)
{
case 0 :

P2OUT = digits[2];
P1OUT = IND2+IND3+IND4;
break;

case 1 :

P2OUT = digits[0];
P1OUT = IND1+IND3+IND4;
break;

case 2 :

P2OUT = digits[1];
P1OUT = IND2+IND1+IND4;
break;

case 3 :

P2OUT = digits[4];
P1OUT = IND2+IND3+IND1;
break;
}
NUM++;
if (NUM == 4) NUM = 0;

}

Последний код это я просто выводил цифры 2014.
Нажмите для просмотра прикрепленного файла
rezident
Вы формулу для прямой линии из курса средней школы помните: y=k*x? С учетом смещения относительно нуля формула выглядит так y=k*(x-x0)+y0, где y и x это значения функции и ее аргумента, а y0 и x0 - смещения по осям Y и X относительно нуля.
В применении к вашему случаю формула будет выглядеть так
F=K*(ADCvalue-ADCmin)+Fmin, где K=(Fmax-Fmin)/(ADCmax-ADCmin)
F это значение частоты, которое соответствует текущему значению ADCvalue.
Cruz21
Цитата(rezident @ Jan 7 2014, 22:00) *
Вы формулу для прямой линии из курса средней школы помните: y=k*x? С учетом смещения относительно нуля формула выглядит так y=k*(x-x0)+y0, где y и x это значения функции и ее аргумента, а y0 и x0 - смещения по осям Y и X относительно нуля.
В применении к вашему случаю формула будет выглядеть так
F=K*(ADCvalue-ADCmin)+Fmin, где K=(Fmax-Fmin)/(ADCmax-ADCmin)
F это значение частоты, которое соответствует текущему значению ADCvalue.

спасибо за инфу, но а как вывести эти значения потом в индикатор?
rezident
Какой именно момент вас напрягает: преобразование числа в строку символов или вывод строки на 7-сегментный индикатор?
Универсальный способ преобразования с помощью функции (s)printf. Недостатки его: требуется использование библиотеки и возможен большой расход стека, особенно если вы будете преобразовывать float в строку. Достоинства: это стандартная Си-ная функция.
Cruz21
Цитата(rezident @ Jan 7 2014, 22:00) *
F=K*(ADCvalue-ADCmin)+Fmin, где K=(Fmax-Fmin)/(ADCmax-ADCmin)
F это значение частоты, которое соответствует текущему значению ADCvalue.

а откуда мне получить значения ADCvalue, ADCmin, Fmax, Fmin, ADCmax, ADCmin?
У меня есть одно 16-значное число занесенный в ADC10MEM.
rezident
Цитата(Cruz21 @ Jan 8 2014, 13:51) *
а откуда мне получить значения ADCvalue, ADCmin, Fmax, Fmin, ADCmax, ADCmin?
У меня есть одно 16-значное число занесенный в ADC10MEM.

ADCvalue это и есть текущее значение, считываемое из ADC10MEM. Остальные значения это константы, которые определяют соответствие диапазона регулировки диапазону частоты. Геометрически это координаты двух точек (X1=ADCmin, Y1=Fmin и X2=ADCmax, Y2=Fmax) на плоскости, через которые вы проводите прямую линию. Прямая линия собственно и определяет характеристику зависмости частоты от регулировки. Если же вам нужна, допустим, не линейная, а логарифмическая зависимость, то и формула для вычисления частоты должна включать в себя логарифм.
У вас регулятор включен как потенциометр между потенциалами питания МК или как-то по-другому? Если как потенциометр промеж питания, то в настрйках АЦП выберите опору тоже питание МК (SREFx=0). Тогда крайние точки для оси абцисс у вас будут ADCmin=0x0000, ADCmax=0x03FF. Тогда значеням Fmin и Fmax вы задаете соответствие крайним положениям движка регулятора. Только не задавайте Fmin=0Гц. Частоту 0Гц на конечном интервале времени определить крайне затруднительно sm.gif
Cruz21
Цитата(rezident @ Jan 8 2014, 17:23) *
ADCvalue это и есть текущее значение, считываемое из ADC10MEM. Остальные значения это константы, которые определяют соответствие диапазона регулировки диапазону частоты. Геометрически это координаты двух точек (X1=ADCmin, Y1=Fmin и X2=ADCmax, Y2=Fmax) на плоскости, через которые вы проводите прямую линию. Прямая линия собственно и определяет характеристику зависмости частоты от регулировки. Если же вам нужна, допустим, не линейная, а логарифмическая зависимость, то и формула для вычисления частоты должна включать в себя логарифм.
У вас регулятор включен как потенциометр между потенциалами питания МК или как-то по-другому? Если как потенциометр промеж питания, то в настрйках АЦП выберите опору тоже питание МК (SREFx=0). Тогда крайние точки для оси абцисс у вас будут ADCmin=0x0000, ADCmax=0x03FF. Тогда значеням Fmin и Fmax вы задаете соответствие крайним положениям движка регулятора. Только не задавайте Fmin=0Гц. Частоту 0Гц на конечном интервале времени определить крайне затруднительно sm.gif

как написать подпрограмму, которая бы из 16 системы счисления преобразовывала в 10 систему счисления в с++. Я сам не особо силен в C++
rezident
Цитата(Cruz21 @ Jan 12 2014, 15:24) *
как написать подпрограмму, которая бы из 16 системы счисления преобразовывала в 10 систему счисления в с++. Я сам не особо силен в C++

Про функцию форматированного вывода sprintf я вам еще в сообщении №4 написал. Вы вообще читаете, что вам пишут-то? wink.gif
Cruz21
Цитата(rezident @ Jan 12 2014, 15:14) *
Про функцию форматированного вывода sprintf я вам еще в сообщении №4 написал. Вы вообще читаете, что вам пишут-то? wink.gif

дело в том что не хочу я использовать функцию форматированного вывода sprintf. Можно же и без него обойтись?
rezident
Цитата(Cruz21 @ Jan 12 2014, 16:38) *
дело в том что не хочу я использовать функцию форматированного вывода sprintf. Можно же и без него обойтись?

А операцию деления тоже нельзя использовать? rolleyes.gif Тогда, пожалуй, остается цикл с последовательным вычитанием чисел 10 в N-й степени (100000, 10000, 1000, 100, 10 и т.п.) laughing.gif
PRidon
http://electronix.ru/forum/index.php?showtopic=117438
Тут посмотри.
Если сухо: на MSP быстрее всего вычитание степеней 10.
akl
Можно провести преобразование так
Код
BIN_BCD:
    MOV        RESULT_H,R8
    MOV        RESULT_L,R9

BIN_BCD0:
     MOV #32,R15

     CLR R10
     CLR R11
     CLR R12
     CLR R13
     CLR R14
BIN_BCD1:
    RLA        R9
    RLC        R8

     DADD     R14,R14
     DADD     R13,R13
     DADD     R12,R12
     DADD     R11,R11
     DADD     R10,R10

     DEC R15
     JNZ BIN_BCD1

Взято из примеров старой книги по MSP430. Как это будет выглядеть на Цэ, не знаю.
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Invision Power Board © 2001-2025 Invision Power Services, Inc.