Перейти к содержанию

Отрицательные температуры ds18b20


Рекомендуемые сообщения

Ребята, я вот делаю термометр на atmega8 + ds18b20+ семисегментник, и столкнулся с такой вот проблемкой . Как реализовать вывод отрицательной температуры???

В моем коде, температуры те что выше 0  отображаются нормально, а вот те, что ниже , какие-то каркозяблы........ 

#include <mega8.h>
#include <delay.h>


#asm
   .equ __w1_port=0x15 ;PORTC
   .equ __w1_bit=2
#endasm

#include <1wire.h>
#include <ds18b20.h>
#define MAX_DS1820 8


 unsigned char rom_codes[MAX_DS1820][9]; //переменная хранения rom кодов
 unsigned char temp[8]; // переменная для хранения значений температуры
 unsigned char devices; // переменная для устройств
 
 
flash char number[] = 
{
  0x3f, //0
  0x06, //1
  0x5b, //2
  0x4f, //3   
  0x66, //4
  0x6d, //5 
  0x7d, //6
  0x07, //7   
  0x7f, //8
  0x6f,  //9 
  0x00,  //blank
  0x40   // minus
};
   

//числа для вывода на индикатор
volatile unsigned char data1 = 0;
volatile unsigned char data2 = 0;
volatile unsigned char data3 = 0;
volatile unsigned char data4 = 0;
volatile unsigned char data5 = 0;
volatile unsigned char data6 = 0;





void temperature(void)
{
 temp[0]=ds18b20_temperature(&rom_codes[0][0]);
 temp[1]=ds18b20_temperature(&rom_codes[1][0]); 
       
        if (temp>1000){
            temp[0]=4096-temp[0];               
            temp[0]=-temp[0];
            data4 = 11;
               }
            
            if(temp>1000){
            temp[1]=4096-temp[1];
            temp[1]=-temp[1];
            data1 = 11;
              } 
              
    data6 = temp[0]%10;
    data5 = temp[0]/10;
    data4 = 10;
    data3 = temp[1]%10;
    data2 = temp[1]/10;
    data1 = 10;
    }

   
    
    
void main( void )
{

  //порт, к которому подкл. сегменты
  PORTD = 0xff;
  DDRD = 0xff;
  
  PORTC=0xFF;
  DDRC=0x00;

  
  //порт, к которому подкл. катод
  PORTB = 0xff;
  DDRB |= (1<<5)|(1<<4)|(1<<3)|(1<<2)|(1<<1)|(1<<0); 
  
  //инициализация таймера Т0 
  TCCR0 = (1<<CS02)|(0<<CS01)|(1<<CS00);
  TCNT0 =  0xE6;
  
  // Timer(s)/Counter(s) Interrupt(s) initialization
TIMSK=(0<<OCIE2) | (0<<TOIE2) | (0<<TICIE1) | (0<<OCIE1A) | (0<<OCIE1B) | (0<<TOIE1) | (1<<TOIE0);

// Оприделение устройств
devices=w1_search(0xf0,rom_codes);
 
  #asm("sei")
  
 
  
  while(1){
   
   temperature();
        
  }  
}

//прерывания таймера Т0 - вывод на индикатор
interrupt [TIM0_OVF] void Timer0Ovf(void)
{
  static unsigned char count = 0;
  TCNT0 = 0xE6;
  
   //гасим все разряды
   PORTB |= (1<<5)|(1<<4)|(1<<3)|(1<<2)|(1<<1)|(1<<0);
   
   //зажигаем следующий разряд
   if (count == 0) {
      PORTD = number[data1];
      PORTB = ~(1<<0);
   }
   if (count == 1) {
      PORTD = number[data2];
      PORTB = ~(1<<1);
   }
   if (count == 2) {
      PORTD = number[data3];
      PORTB = ~(1<<2);
    }  
   if (count == 3) {
      PORTD = number[data4];
      PORTB = ~(1<<3);
      }
   if (count == 4) {
      PORTD = number[data5];
      PORTB = ~(1<<4);
      }
   if (count == 5) {
      PORTD = number[data6];
      PORTB = ~(1<<5);
      }     
    
    count++;
    if (count == 6) count = 0;

}

 

Ссылка на комментарий
Поделиться на другие сайты

Реклама: ООО ТД Промэлектроника, ИНН: 6659197470, Тел: 8 (800) 1000-321

29 минут назад, Zodiac21 сказал:

unsigned char temp

А температура с датчика, является знаковой переменной.
 

Errare humanum est. Коли людЯм позволено, что же о нас то говорить!
 

Ссылка на комментарий
Поделиться на другие сайты

20% скидка на весь каталог электронных компонентов в ТМ Электроникс!

Акция "Лето ближе - цены ниже", успей сделать выгодные покупки!

Плюс весь апрель действует скидка 10% по промокоду APREL24 + 15% кэшбэк и бесплатная доставка!

Перейти на страницу акции

Реклама: ООО ТМ ЭЛЕКТРОНИКС, ИНН: 7806548420, info@tmelectronics.ru, +7(812)4094849

Выбираем схему BMS для корректной работы литий-железофосфатных (LiFePO4) аккумуляторов

 Обязательным условием долгой и стабильной работы Li-FePO4-аккумуляторов, в том числе и производства EVE Energy, является применение специализированных BMS-микросхем. Литий-железофосфатные АКБ отличаются такими характеристиками, как высокая многократность циклов заряда-разряда, безопасность, возможность быстрой зарядки, устойчивость к буферному режиму работы и приемлемая стоимость. Но для этих АКБ, также как и для других, очень важен контроль процесса заряда и разряда, а специализированных микросхем для этого вида аккумуляторов не так много. Инженеры КОМПЭЛ подготовили список имеющихся микросхем и возможных решений от разных производителей. Подробнее>>

Реклама: АО КОМПЭЛ, ИНН: 7713005406, ОГРН: 1027700032161

Только что, o_l_e_g сказал:

А температура с датчика, является знаковой переменной.
 

К тому же 16-битной...

http://arv.radioliga.com/content/view/106/49/ - вот здесь я писал о том, как можно выводить числа на семисегментник, в том числе и с минусом.

Если забанить всех, кто набрался смелости думать независимо, здорово будет на форуме - как на кладбище: тишина, птички поют...

Ссылка на комментарий
Поделиться на другие сайты

Будьте добры,  подскажите как на примере моего кода это организовать.....

Вот я подключил эту функцию, а как дальше быть?

#include <mega8.h>
#include <delay.h>

 
// DS1307 Real Time Clock functions
#include <ds1307.h>

#asm
   .equ __w1_port=0x15 ;PORTC
   .equ __w1_bit=2
#endasm

#include <1wire.h>
#include <ds18b20.h>
#define MAX_DS1820 8


 unsigned char rom_codes[MAX_DS1820][9]; //переменная хранения rom кодов
 int temp[8]; // переменная для хранения значений температуры
 unsigned char devices; // переменная для устройств
 
  
            #define DIG_BASE  10 /* основание системы считсления для перевода */
			#define MAX_SIZE 6 /* максимальное число выводимых символов */ 
			#define SPACE_CHAR  ' ' /* символ "пустого" места */
			#define NEG_CHAR '-' /* символ "минус" */
 
  unsigned char out[MAX_SIZE]; // выходной массив символов (экранная область)
  
  
flash char number[] = 
{
  0x3f, //0
  0x06, //1
  0x5b, //2
  0x4f, //3   
  0x66, //4
  0x6d, //5 
  0x7d, //6
  0x07, //7   
  0x7f, //8
  0x6f,  //9 
  0x00,  //blank
  0x40   // minus
};
   

//числа для вывода на индикатор
volatile unsigned char data1 = 10;
volatile unsigned char data2 = 0;
volatile unsigned char data3 = 0;
volatile unsigned char data4 = 10;
volatile unsigned char data5 = 0;
volatile unsigned char data6 = 0;





void temperature(void)
{
 temp[0]=ds18b20_temperature(&rom_codes[0][0]);
 temp[1]=ds18b20_temperature(&rom_codes[1][0]); 
       
        if (temp>1000){
            temp[0]=4096-temp[0];               
            temp[0]=-temp[0];            
               }
            
            if(temp>1000){
            temp[1]=4096-temp[1];
            temp[1]=-temp[1];
              } 
              
    data6 = temp[0]%10;
    data5 = temp[0]/10;
   
    data3 = temp[1]%10;
    data2 = temp[1]/10;
    
    
    }

  
  void s_trim_convert(int NUM){
			 int i, m, sign = 0;
			 if(NUM <0){
			 // если число отрицательное
			  sign = 1; // установим признак наличия знака
			  NUM *= -1; // а само число возьмем по модулю
			 }
			 // выводим уже положительное число
			 i=MAX_SIZE-1;
			 do{
			  // цикл заполнения выходного массива СПРАВА НАЛЕВО
			  m = NUM % DIG_BASE; // находим остаток от деления числа на основание
			  if((NUM==0)&&(i!=(MAX_SIZE-1)))
			   break; // закончим цикл вывода числа
			  else
			   out[i] = number[m]; // иначе выводим символ нужной ЦИФРЫ
			  NUM /= DIG_BASE;  // уменьшаем число в DIG_BASE раз
			 } while (--i >= 0);
			 // число выведено, проверяем свободное место и выводим при необходимости знак
			 if (i < 0) return; // места не хватает - выход
			 if (sign) out[i--] = NEG_CHAR; // выводим знак, если нужно
			 for(; i>=0;i--) out[i] = SPACE_CHAR; // очищаем незначащие позиции 
			}
 
    
    
void main( void )
{

  //порт, к которому подкл. сегменты
  PORTD = 0xff;
  DDRD = 0xff;
  
  PORTC=0xFF;
  DDRC=0x00;

  
  //порт, к которому подкл. катод
  PORTB = 0xff;
  DDRB |= (1<<5)|(1<<4)|(1<<3)|(1<<2)|(1<<1)|(1<<0); 
  
  //инициализация таймера Т0 
  TCCR0 = (1<<CS02)|(0<<CS01)|(1<<CS00);
  TCNT0 =  0xE6;
  
  // Timer(s)/Counter(s) Interrupt(s) initialization
TIMSK=(0<<OCIE2) | (0<<TOIE2) | (0<<TICIE1) | (0<<OCIE1A) | (0<<OCIE1B) | (0<<TOIE1) | (1<<TOIE0);

// Оприделение устройств
devices=w1_search(0xf0,rom_codes);
 
  #asm("sei")
  
 
  
  while(1){
  s_trim_convert(temp[0]); 
   temperature();
        
  }  
}

//прерывания таймера Т0 - вывод на индикатор
interrupt [TIM0_OVF] void Timer0Ovf(void)
{
  static unsigned char count = 0;
  TCNT0 = 0xE6;
  
   //гасим все разряды
   PORTB |= (1<<5)|(1<<4)|(1<<3)|(1<<2)|(1<<1)|(1<<0);
   
   //зажигаем следующий разряд
   if (count == 0) {
      PORTD = number[data1];
      PORTB = ~(1<<0);
   }
   if (count == 1) {
      PORTD = number[data2];
      PORTB = ~(1<<1);
   }
   if (count == 2) {
      PORTD = number[data3];
      PORTB = ~(1<<2);
    }  
   if (count == 3) {
      PORTD = number[data4];
      PORTB = ~(1<<3);
      }
   if (count == 4) {
      PORTD = number[data5];
      PORTB = ~(1<<4);
      }
   if (count == 5) {
      PORTD = number[data6];
      PORTB = ~(1<<5);
      }     
    
    count++;
    if (count == 6) count = 0;

}

 

Ссылка на комментарий
Поделиться на другие сайты

Получилось сделать следующим образом: 

void temperature(void)
{
 temp[0]=ds18b20_temperature(&rom_codes[0][0]);
 temp[1]=ds18b20_temperature(&rom_codes[1][0]); 
       
        if (temp[0]<0){               
            temp[0]=~temp[0]+1;
            data4=11;            
               }
            
            if(temp[1]<0){
            temp[1]=~temp[1]+1;
            data1=11;
              } 
              
    data6 = temp[0]%10;
    data5 = temp[0]/10;
   
    data3 = temp[1]%10;
    data2 = temp[1]/10;
    
    
    }

То есть проинвертировав значения переменной temp и прибавив 1.......

Вроде всё работает........ 

Снимок1.png

Ссылка на комментарий
Поделиться на другие сайты

Присоединяйтесь к обсуждению

Вы можете написать сейчас и зарегистрироваться позже. Если у вас есть аккаунт, авторизуйтесь, чтобы опубликовать от имени своего аккаунта.
Примечание: Ваш пост будет проверен модератором, прежде чем станет видимым.

Гость
Unfortunately, your content contains terms that we do not allow. Please edit your content to remove the highlighted words below.
Ответить в этой теме...

×   Вставлено с форматированием.   Восстановить форматирование

  Разрешено использовать не более 75 эмодзи.

×   Ваша ссылка была автоматически встроена.   Отображать как обычную ссылку

×   Ваш предыдущий контент был восстановлен.   Очистить редактор

×   Вы не можете вставлять изображения напрямую. Загружайте или вставляйте изображения по ссылке.

Загрузка...
  • Последние посетители   0 пользователей онлайн

    • Ни одного зарегистрированного пользователя не просматривает данную страницу
  • Сообщения

    • Попробуй еще раз сакцентировать внимание. В рамках моего вопроса, рассматривается ТОЛЬКО способ крепления разьема с ПРЯМЫМИ ногами, ВДОЛЬ платы. Другие способы не подойдут. Никак. Совсем. Их нет смысла советовать. И основной вопрос "как разместить переходные отверстия, что бы максимально крепко пролить оловом ноги разьема?". Желательно без клея. Пока из дельного насоветовали только что-то типа Оловянных Клепок, через овтерстия максимально большого проходного сечения  
    • ДБ функция логарифмическая, а на твоём спектралабе шкала дб линейная, почему??? Радиотехник ты неплохой, а вот метролог из тебя левый. Учи матчасть и не зас_рай  тему своим офтопом.
    • правильно мыслишь... скорее всего надо увеличить сопротивление R1 до 2-3 кОм.. (или R3)..
    • @r9o-11 Я про перемотку трансформатора. Не, возможно если магнитопроводов под рукой запас, провода запас, станочек есть, рука набита - тогда да, это будет быстрее и проще. Но мне кажется что в современных реалиях добыть два импульсных БП подходящей мощности всё-же легче. А человеку далёкому от электроники - гораздо легче. 
    • Нужен электронный реостат для нагрузки блоков питания. Т.е. я собираю не электронную нагрузку, а электронный реостат. Здесь нет стабилизации ни тока, ни напряжения. Слева - подается питающее напряжение, справа - подключается исследуемый блок питания. Когда движок потенциометра внизу транзистор закрыт и ток через него практически не идет. Сдвигая движок, транзистор приоткрывается и начинает пропускать ток. Тем самым создаем нагрузку для исследуемого блока питания. Но возник вопрос. Одновременно оба напряжения - питающее и исследуемое подаваться не будут. И получается, что в какой то момент времени транзистор будет работать с оборванной базой или с оборванным коллектором. Допустимо ли это? Не повредится ли он?
    • Вот это плохие новости. Я был готов к такому повороту, но у меня всегда был уверенный запуск с ППГ сердечниками, термообработанными без наложения поля (с индексом N). ППГ подходят лучше всего, особенно для мощных ключей – обеспечивают наименьшие динамические потери, с ними можно увеличивать номинал балластного резистора в цепи ПОС. Еще лучше с отжигом в продольном поле, но от однополярного динисторного пускача, такие точно будут запускаться с вероятностью ровно 50%. При желании  выкрутиться можно и с очень высокой прямоугольностью, если запускающие импульсы будут поочередно-разнополярными, в этом случае обеспечен 100% запуск независимо от того, в каком квадранте находилась остаточная индукция перед запуском.
×
×
  • Создать...