Проблема с одним проводом - PullRequest
4 голосов
/ 15 декабря 2009

Мне нужна ваша квалифицированная помощь! Я программирую на C ++, использую PIC 18F87J50 и пытаюсь подключить DS18B20 к моему порту H0!

Я думаю, что мое основное программирование правильное, так что ... проблема у меня (кажется, у меня есть), при выполнении команды ПЗУ я ищу 64-битный КОД ПЗУ.

Первый байт должен сказать мне, к какому семейству принадлежит компонент (28h). Следующие 48 бит должны дать мне уникальный серийный номер только для этого компонента. Последний используется для CRC.

Правильно ли я думаю, когда делаю так:

void Device_ID( uint8_t command ){
    uint8_t ROM_CODE[8]; // 1 byte CRC, 6 bytes SERIAL, 1 byte Family code
    uint8_t loop;
    static char container[8];

    OW_reset_pulse();
    OW_write_byte( command );

    for(loop = 0; loop < 8; loop++)     // 1 byte in per time = 64-bits
    {
        ROM_CODE[loop] = OW_read_byte(); 
    }

    HexToStrWithZeros(ROM_CODE[0], container);
    Display_Cls();
    Display_StringAt ("Family Code: ",5,6);
    Display_Buffer (container);
}

Если я запрашиваю код в ROM_CODE [1-6], я должен получить уникальный номер ?? не должен ли я ??

С уважением!

Ответы [ 2 ]

2 голосов
/ 15 декабря 2009

Что ж, лучший способ получить доступ к серийному номеру - это, вероятно, скопировать его в отдельный буфер, используя strncpy .

#include <string.h>

...

char family;
char serial[7]; // Extra byte for null terminator
char checksum;

...

family = ROM_CODE[0];

strncpy(serial, &ROM_CODE[1], 6);
serial[6] = '\0';

checksum = ROM_CODE[7];

...

&ROM_CODE[1] предназначен для получения адреса 2-го элемента в ROM_CODE. ROM_CODE+1 тоже может сработать, но мой C - ржавый налет.

Нуль ('\ 0') добавляется в конце, поскольку C использует строки с нулевым символом в конце . Это обеспечит совместимость с подпрограммами библиотеки C и обычно используемыми идиомами языка Си.

Вы также можете получить к нему доступ непосредственно из массива. Но с этим будет сложнее работать, и вряд ли оно того будет стоить, если вам действительно не понадобятся эти 6 байтов памяти.

В зависимости от того, насколько сложным является ваше приложение, вы можете захотеть обернуть это в классе. Передайте 8-символьный буфер в конструктор и затем используйте методы, такие как getFamily() / getSerial(), чтобы получить необходимую информацию.

Тем не менее, для очень простого приложения достаточно много дополнительного кода, чтобы упростить то, что уже очень легко управляемо.

0 голосов
/ 26 декабря 2009

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

/****************************************************************************
* temperature.h
****************************************************************************/

#ifndef TEMP_H
#define TEMP_H

extern double read_temp ( void );
extern   void start_temp( void );
extern   void Device_ID ( void );

#endif

/****************************************************************************
* temperature.c
****************************************************************************/

void     reset_ow(void);
void     write_ow(uint8_t  b);
uint8_t  read_ow (void);

#define OW_TEMP_SIG LATHbits.LATH0
#define OW_TEMP_TRIS TRISHbits.TRISH0
#define OW_TEMP_SIG_IN PORTHbits.RH0
#define DIR_OUT 0
#define DIR_IN 1


void Device_ID( void )
{
    uint8_t loop; 
   uint8_t family; 
   uint8_t checksum; 
   uint8_t ROM_CODE[8];         // 1 byte CRC, 6 bytes SERIAL, 1 byte Family code

   reset_ow();
   write_ow(0x33); // READ ROM COMMAND DS18B20

    for(loop = 0; loop < 8; loop++) // 1 byte in per time = 64-bits
    {
        ROM_CODE[loop] = read_ow();
    }

    family = ROM_CODE[0];
    checksum = ROM_CODE[7];

  // add extra code to handle code
}

void start_temp(void) 
{
   uint8_t i;

   OW_TEMP_SIG=1;
   OW_TEMP_TRIS=DIR_OUT;
   for ( i=0;i<100;i++)
   {
       Delay_us(100);
   }
   reset_ow();
   write_ow(0xcc); // skip rom
   write_ow(0x44); // start t conv
}

double read_temp(void) 
{
   double temp=0;
   S16 itemp;

   reset_ow();
   write_ow(0xcc); // skip rom
   write_ow(0xbe); // read scratch pad
   itemp=read_ow();
   itemp|=(S16)read_ow()<<8;

   temp = itemp*(0.0625);
   OW_TEMP_TRIS=DIR_IN;
   OW_TEMP_SIG=1;
   return temp; 
}


void reset_ow(void)
{
   OW_TEMP_TRIS=DIR_OUT;
   OW_TEMP_SIG=0;
   Delay_us(250);
   Delay_us(250);
   OW_TEMP_TRIS=DIR_IN;
   OW_TEMP_SIG=1;
   Delay_us(250);
   Delay_us(250);
}

void write_ow(uint8_t b)
{
   uint8_t i;

   OW_TEMP_SIG=1;
   OW_TEMP_TRIS=DIR_OUT;
   for ( i=0;i<8;i++)
   {
      OW_TEMP_SIG=0;
      if ( b & 0x01 )
      {
         Delay_us(10);
         OW_TEMP_SIG=1;
      }
      Delay_us(70);
      OW_TEMP_SIG=1;
      Delay_us(10);
      b >>= 1;
   }
   OW_TEMP_TRIS=DIR_IN;
   OW_TEMP_SIG=1;
}

uint8_t read_ow(void)
{
   uint8_t b=0;
   uint8_t m;
   uint8_t i;

   m=1;
   for ( i=0;i<8;i++)
   {
      OW_TEMP_SIG=1;
      OW_TEMP_TRIS=DIR_OUT;
      OW_TEMP_SIG=0;        
      Delay_us(8);
      OW_TEMP_TRIS=DIR_IN;
      OW_TEMP_SIG=1;
      Delay_us(15);

      if ( 1 == OW_TEMP_SIG_IN )
      {
         b |= m;
      }
      m <<=1;
      Delay_us(60);
   }
   OW_TEMP_TRIS=DIR_IN;
   OW_TEMP_SIG=1;
   return b;
}
...