Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: прог. реализ. AVR+adc124s101 подкл. через SPI
Форум разработчиков электроники ELECTRONIX.ru > Микроконтроллеры (MCs) > AVR
Чип-Хрум
Нужна программная реализация получения 12 бит от ADC124s101 на Atmega32 .
Пробывал опрашивать как на прямую так и с помощью прерывания ,по моим подсчетам
эта микросхема должна выдавать все 12 бит через каждый мегахерц.
А на самом деле получается около раза в секунду.
Выводится все на жки в таком порядке:
на одной знакопозиции
0909
4b4b
0909
4a4a
0909
4949
0909
4949
Короче сейчас дело не в точности показаний а в том чтобы вывести 0949 за место 09094949.
Может я че в програмке напутал Кто силен подскажите!
#include <inttypes.h>
#include <avr/io.h>
#include <avr/interrupt.h>
#include <stdlib.h>
#include <util/delay.h>
#include <stdio.h>
#include <avr/eeprom.h>

#define bitset(var,bitno) ((var) |= 1 << (bitno))
#define bitclr(var,bitno) ((var) &= ~(1 << (bitno)))

char dat0=0,dat1=0; //переменные, 8бит для dat0 и 4бита для dat1
char revers=0;

ISR(SPI_STC_vect) // вектор прерывания для окончания посылки байта
{

while(revers==0) // что-то типа реверса за одно прохождение
{ // опрашавается одна часть 12 бит
dat0 = SPDR; //
revers=1; //
} //
//
while(revers==1) //
{ //
dat1 = SPDR;
revers=0;
}

bitset(SREG,7);
SPDR = 0x00;
}

int main(void)
{

DDRB=0xb0; // настройка порта b
SPCR=0xd3; // настройка spi

SPDR = 0x00;
bitset(SREG,7); // включаем прерывание

while(1) // вечный цикл

return (0);

}
defunct
А если в лоб, без аппаратного SPI?

Код
#define set_cs()    (PORTB |= (1 << CS))
#define clr_cs()    (PORTB &= ~(1 << CS))
#define set_scl()   (PORTB |= (1 << SCL))
#define clr_scl()   (PORTB &= ~(1 << SCL))
#define get_miso()  ((PINB & (1 << MISO)) ? 1 : 0)

short Read12Bits(void)
{
    int i;
    short result = 0;

    clr_cs();
    for( i = 0; i < 12; i++)
    {
         clr_scl();
         result <<= 1;
         set_scl();
         result |= get_miso();
    }
    set_cs();
    return result;
}
Чип-Хрум
smile.gif Надо заметить идея заслуживающая уважения a14.gif !
Но дело в том что перед тем как получить 12 бит из ADC, нужно 8бит отправить.
Я в си новечек раньше только на ассемблере писал програмки.
Короче теперь нужен цикл отправки переменной char по тому же SPI.
И подскажите где правильно научиться битами управлять в переменных,
а то пролистал всю мукулатуру а дельного ничего нет.
Kolia
Цитата
ISR(SPI_STC_vect) // вектор прерывания для окончания посылки байта
{

while(revers==0) // что-то типа реверса за одно прохождение
{ // опрашавается одна часть 12 бит
dat0 = SPDR; //
revers=1; //
} //
//
while(revers==1) //
{ //
dat1 = SPDR;
revers=0;
}

bitset(SREG,7);
SPDR = 0x00;
}


Не правильно описано работа с прерыванием. Код dat0 = SPDR; и dat1 = SPDR; выполняются за каждый проход (т.е dat0==dat1). Соответственно появляются ошибки при отображении данных.
Должно быть что-то вроде этого.

Код
ISR(SPI_STC_vect) // вектор прерывания для окончания посылки байта
{

bitclr(SPCR,7);

if(SPSR & 0x80){//SPIF
bitclr(SPSR ,7);

if(revers==0) // что-то типа реверса за одно прохождение
{ // опрашавается одна часть 12 бит
dat0 = SPDR; //
revers=1; //
}
else
{ //
dat1 = SPDR;
revers=0;
}
}
else{
bitclr(SPSR ,6);//WCOL=0;
}


SPDR = 0x00;
bitset(SPCR,7);
}
Чип-Хрум
Ошибка и в правду была моя , спасибо за подсказку но всетаки дело не в этом
а втом что этот adc имеет 4 входа на каждый я должен сделать по два запроса ,
в сумме получается что за восемь прохождений я получу четыре байта.
Многова-то получается да и с данными не все так просто по моему аппаратная
реализация не надежна (я оговариваюсь в данном случае ) нужна
программная, постараюсь обьяснить почему .
В datasheet описывается что за один запрос я должен получить 12 бит
заметьте не 16.
Т.Е. я посылаю к примеру 00 а он мнен отвечает fff. И ничего лишнего.
Вопрос не снят с повестки дня помогите кто чем может.
Нужен цикл отправки переменной char по тому же SPI в программной реализации ,а не в аппоратной!!!
И подскажите где правильно научиться битами управлять в переменных,
а то пролистал всю мукулатуру а дельного ничего нет.
WHALE
char a;
a|=0x80; //установить старший бит
a&=0x7F //сбросить старший бит
Kolia
Цитата(Чип-Хрум @ Apr 2 2007, 19:42) *
Нужен цикл отправки переменной char по тому же SPI в программной реализации ,а не в аппоратной!!!


Для вашей задачи нет различия между программным и аппаратным SPI. Просто аппаратный намного проше, быстрее и занимеет меньше памяти.
Микросхема ADC124s101 работает с пакетами данных длиной 16 бит.
Временную диаграмму можно посмотреть в Datasheete. Вначале отправляется упрявляющий байт, а затем нулевой байт. И соответственно в этой посылке принимаются 16 бит данных (по 2 байта), 12 бит которых являются информативными.
Что-бы микросхема выдала напряжение какого-либо канала неабходимо закинуть ей в первом пакете 2 байта с номером канала (например 0x08 0x00 - ADD0), затем ждем преобразования ADC124s101 (примерно 50 мкс), и читаем полученный результат следующей посылкой (0x08, 0x00 - здесь можно указать следующий канал ADC - в данном случае это ADD0), затем ждем преобразования ... и т.д.

Вот примерный код

Код
int ReadFromAdc(unsigned char Chanel){
int OUT;
if(Chanel>2) return -1;

CS=0; //чип селект на микросхеме
SPDR =( (0x01<<3)<<Chanel);
while ((SPSR & 0x80) == 0); // wait SPIF
SPSR&=~0x80;
OUT=SPDR;
OUT=OUT<<8;

SPDR = 0x00;
while ((SPSR & 0x80) == 0); // wait SPIF
SPSR&=~0x80;
OUT |=SPDR;

CS=1; //чип селект на микросхеме

return OUT;
}

main(void)
int ADC1;
......//настройка

bitclr(SREG,7); //не включаем прерывание


ReadFromAdc(1); // настройка на канал 1
delay_mks(50);
while(1) // вечный цикл
{

ADC1=ReadFromAdc(1);  
delay_mks(50);
}

Если вас все же интересует программная реализация, то могу выложить.
Чип-Хрум
Теперь проблемма в другой части , нашел термодатчик lm335 он вроде-бы
подходит для этой цели(подключение к ацп) в даташит описан способ
подключения 3 последовательно +сопротивление .
Вопрос : каким образом этот набор полупроводников отколибровать ,
без загрубления градации характеристик на ноль.
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Invision Power Board © 2001-2025 Invision Power Services, Inc.