Попытка понять, что Microchip PIC16LF15344 выбирает периферийные контакты для I2C - PullRequest
0 голосов
/ 03 января 2019

Я просматривал таблицу данных Microchip PIC16LF15344 для выбора периферийных выводов, либо я неправильно читаю, либо в документе обнаружены ошибки и несоответствия. Я хотел бы знать, использовал ли кто-либо еще это устройство и может ли подтвердить мою интерпретацию.

Я пытаюсь написать код для PIC16LF15344, чтобы использовать интерфейс I2C. Мне удалось написать I2C код для PIC16LF1822, который работает нормально, но я до сих пор не смог заставить его работать на PIC16LF15344, и в листе данных есть некоторая запутанная документация, которая мне бы хотелось чтобы очиститься.

Вот ссылка на таблицу данных .

Описания pinout для PIC16LF15344 показывают, что функция I2C SDA может быть присвоена RC1 или RB6. Аналогично, I2C SCL может быть выделено для RC0 или RB4. Но в Section 15.3 Bidirectional Pins есть примечание:

Функции I2C SCLx и SDAx можно переназначить через PPS. Однако только пинты RB1, RB2, RC3 и RC4 имеют специальные входные буферы I2C и SMBus (режим I2C отключает INLVL и устанавливает пороговые значения, специфичные для I2C). Если функции SCLx или SDAx отображаются на какой-либо другой вывод (кроме RB1, RB2, RC3 или RC4), входные буферы общего назначения TTL или ST ( вместо этого будет использоваться конфигурация на основе INLVL настройки регистра). Поэтому в большинстве приложений рекомендуется только сопоставить функции выводов SCLx и SDAx с выводами RB1, RB2, RC3 или RC4.

Проблема заключается не только в том, что заметка конфликтует с описаниями в таблицах распределения контактов, она также ссылается на контакты RB1 и RB2, которые нигде не отображаются в таблицах распределения контактов, т. Е. Они кажется, не существует для этого PIC. Я вижу похожие ссылки на сноски на RB1 и RB2, но они не документированы нигде в теле документа или в таблицах.

Конечно, это должна быть ошибка документации, но я не могу найти текущих ошибок в таблице PIC16LF15324/44, чтобы исправить это. Я правильно читаю?

Я попытался настроить SCL и SDA с PPS на контакты RC0 и RC1, поскольку именно так они подключены к моему PCB, но я не могу получить I2C работает, используя, по сути, то же самое программное обеспечение, которое я использовал для PIC16LF1822. PPS по умолчанию для EUSART TX2 и RX2: RC0 и RC1. Означает ли это, что мне нужно использовать TX2CKPPS и RX2DTPPS для перемещения назначений EUSART в другое место, прежде чем использовать SSP1CLKPPS и SSP1DATPPS для назначения SCL и SDA на RC0 и RC1?

Я буду исследовать другие потенциальные проблемы с моим PCB, но я хочу исправить это, прежде чем отправлять еще один PCB для потрясающего. В любом случае мне нужно внести некоторые изменения, поэтому я мог бы просто подключить SCL и SDA к их PPS значениям по умолчанию в следующей версии и повторить попытку.

Вот код инициализации:

OSCFRQbits.HFFRQ = 0b011; // Set internal HF oscillator frequency to 8 MHz
WPUA   = 0b00111111;    // Enable all weak pull-up resistors on port A
WPUB   = 0b11110000;    // Enable all weak pull-up resistors on port B
WPUC   = 0b11111100;    // Enable all weak pull-up resistors on port C except
                        //    RC0 and RC1 to be used as I2C SCL and SDA
TRISA  = 0b00110000;    // Set RA4 and RA5 as inputs
ANSELA = 0b00110000;    // Set RA4 and RA5 to analog
TRISB  = 0b00110000;    // Set RB6 and RB7 as inputs
ANSELB = 0b11000000;    // Set RB6 and RB7 as analog
TRISC  = 0b11111011;    // Set RC0, RC1, RC3, RC4, RC5, RC6, and RC7 as inputs
ANSELC = 0b11111000;    // Set RC3, RC4, RC5, RC6, and RC7 as analog
TX2CKPPS   = 0b01100;   // Use RB4 for TX2
RX2DTPPS   = 0b01110;   // Use RB6 for RX2
SSP1CLKPPS = 0b10000;   // Use RC0 as SCL
SSP1DATPPS = 0b10001;   // Use RC1 as SDA
SSP1CON1 = 0b00100110;  // SSPEN enabled, WCOL no collision, SSPOV no overflow,
                        // CKP low hold, SSPM I2C slave 7-bit

SSP1CON2 = 0b00000000;  // ACKSTAT received, RCEN disabled, RSEN disabled, 
                        // ACKEN disabled, ACKDT acknowledge, SEN disabled, 
                        // GCEN disabled, PEN disabled

SSP1CON3 = 0b00000000;  // BOEN disabled, AHEN disabled, SBCDE disabled, 
                        // SDAHT 100 ns hold, ACKTIM ackseq, DHEN disabled, 
                        // PCIE disabled, SCIE disabled

SSP1STAT = 0x00;
SSP1BUF  = 0x00;
SSP1MSK  = 0xff;
SSP1ADD  = I2C_SLAVE_ADDR << 1;
PIR3bits.SSP1IF = 0;    // Clear the SSP Interrupt flag
PIE3bits.SSP1IE = 1;    // Enable SSP Interrupts
INTCONbits.GIE = 1;     // Enable global interrupts
INTCONbits.PEIE = 1;    // Enable peripheral interrupts

1 Ответ

0 голосов
/ 03 января 2019

Скорее всего, вам нужно решить несколько проблем.

Все стандартные I2C выводы также имеют аналоговые функции.Убедитесь, что биты ANSB или ANSC, связанные с выводами I2C, настроены для работы в цифровом режиме.

Когда при включении питания выбираются значения по умолчанию для листа данных для входного контакта I2Cназначения выходы I2C по умолчанию не назначены ни одному выводу GPIO.Вам нужно будет разместить I2C выходы для SCL и SDA в правильных PPS отображающих регистрах.

Обратите внимание, что независимо от реализации I2C master или slave и вход, и выходфункции должны быть сопоставлены с одним и тем же выводом.

Это поможет отредактировать ваш вопрос и опубликовать код, который вы используете для инициализации I2C выводов и PPS регистров отображения.

/*
 * File:   main.c
 *                             PIC16LF15354
 *                   +-------------:_:-------------+
 * 10K Pull-Up ->  1 : RE3/MCLR   ANB7/RX2/PGD/RB7 : 28 <> 
 *             <>  2 : RA0/ANA0   ANB6/TX2/PGC/RB6 : 27 <> RX2
 *             <>  3 : RA1/ANA1           ANB5/RB5 : 26 <>
 *             <>  4 : RA2/ANA2           ANB4/RB4 : 25 <> TX2
 *             <>  5 : RA3/ANA3           ANB3/RB3 : 24 <>
 *             <>  6 : RA4/ANA4      ANB2/SDA2/RB2 : 23 <>
 *             <>  7 : RA5/ANA5      ANB1/SCL2/RB1 : 22 <>
 *         GND ->  8 : VSS                ANB0/RB0 : 21 <>
 *             <>  9 : RA7/OSC1/ANA7           VDD : 20 <- 3v3
 *             <> 10 : RA6/OSC2/ANA6           VSS : 19 <- GND
 *        SCL1 <> 11 : RC0/ANC0       ANC7/RX1/RC7 : 18 <>
 *        SDA1 <> 12 : RC1/ANC1       ANC6/TX1/RC6 : 17 <>
 *             <> 13 : RC2/ANC2           ANC5/RC5 : 16 <> 
 *             <> 14 : RC3/SCL1/ANC3 ANC4/SDA1/RC4 : 15 <>
 *                   +-----------------------------:
 *                               DIP-28
 *
 * Created on January 4, 2019, 6:20 PM
 */


// PIC16LF15354 Configuration Bit Settings
#pragma config FEXTOSC = OFF    // External Oscillator mode selection bits (Oscillator not enabled)
#pragma config RSTOSC = HFINT32 // Power-up default value for COSC bits (HFINTOSC with OSCFRQ= 32 MHz and CDIV = 1:1)
#pragma config CLKOUTEN = OFF   // Clock Out Enable bit (CLKOUT function is disabled; i/o or oscillator function on OSC2)
#pragma config CSWEN = ON       // Clock Switch Enable bit (Writing to NOSC and NDIV is allowed)
#pragma config FCMEN = OFF      // Fail-Safe Clock Monitor Enable bit (FSCM timer disabled)
#pragma config MCLRE = ON       // Master Clear Enable bit (MCLR pin is Master Clear function)
#pragma config PWRTE = OFF      // Power-up Timer Enable bit (PWRT disabled)
#pragma config LPBOREN = OFF    // Low-Power BOR enable bit (ULPBOR disabled)
#pragma config BOREN = OFF      // Brown-out reset enable bits (Brown-out reset disabled)
#pragma config BORV = LO        // Brown-out Reset Voltage Selection (Brown-out Reset Voltage (VBOR) set to 1.9V on LF, and 2.45V on F Devices)
#pragma config ZCD = OFF        // Zero-cross detect disable (Zero-cross detect circuit is disabled at POR.)
#pragma config PPS1WAY = OFF    // Peripheral Pin Select one-way control (The PPSLOCK bit can be set and cleared repeatedly by software)
#pragma config STVREN = ON      // Stack Overflow/Underflow Reset Enable bit (Stack Overflow or Underflow will cause a reset)
#pragma config WDTCPS = WDTCPS_31// WDT Period Select bits (Divider ratio 1:65536; software control of WDTPS)
#pragma config WDTE = SWDTEN    // WDT operating mode (WDT enabled/disabled by SWDTEN bit in WDTCON0)
#pragma config WDTCWS = WDTCWS_7// WDT Window Select bits (window always open (100%); software control; keyed access not required)
#pragma config WDTCCS = SC      // WDT input clock selector (Software Control)
#pragma config BBSIZE = BB512   // Boot Block Size Selection bits (512 words boot block size)
#pragma config BBEN = OFF       // Boot Block Enable bit (Boot Block disabled)
#pragma config SAFEN = OFF      // SAF Enable bit (SAF disabled)
#pragma config WRTAPP = OFF     // Application Block Write Protection bit (Application Block not write protected)
#pragma config WRTB = OFF       // Boot Block Write Protection bit (Boot Block not write protected)
#pragma config WRTC = OFF       // Configuration Register Write Protection bit (Configuration Register not write protected)
#pragma config WRTSAF = OFF     // Storage Area Flash Write Protection bit (SAF not write protected)
#pragma config LVP = OFF        // Low Voltage Programming Enable bit (High Voltage on MCLR/Vpp must be used for programming)
#pragma config CP = OFF         // UserNVM Program memory code protection bit (UserNVM code protection disabled)

#include <xc.h>

#define I2C_SLAVE_ADDR 0x00

void main(void) 
{
    WPUA   = 0b00111111;    // Enable all weak pull-up resistors on port A
    WPUB   = 0b11110000;    // Enable all weak pull-up resistors on port B
    WPUC   = 0b11111100;    // Enable all weak pull-up resistors on port C except
                            //    RC0 and RC1 to be used as I2C SCL and SDA
    TRISA  = 0b00110000;    // Set RA4 and RA5 as inputs
    ANSELA = 0b00110000;    // Set RA4 and RA5 to analog
    TRISB  = 0b00110000;    // Set RB6 and RB7 as inputs
    ANSELB = 0b11000000;    // Set RB6 and RB7 as analog
    TRISC  = 0b11111011;    // Set RC0, RC1, RC3, RC4, RC5, RC6, and RC7 as inputs
    ANSELC = 0b11111000;    // Set RC3, RC4, RC5, RC6, and RC7 as analog
#ifdef WRONG_WAY_TO_DO_PPS
    TX2CKPPS   = 0b01100;   // Use RB4 for TX2
    RX2DTPPS   = 0b01110;   // Use RB6 for RX2
    SSP1CLKPPS = 0b10000;   // Use RC0 as SCL
    SSP1DATPPS = 0b10001;   // Use RC1 as SDA
#else
    RB4PPS     = 0x11;      // Assign TX2 output  to RB4
    RX2DTPPS   = 0x0E;      // Assign RB6         to RX2 input
    RC0PPS     = 0x15;      // Assign SCL1 output to RC0
    SSP1CLKPPS = 0x10;      // Assign RC0         to SCL1 input
    RC1PPS     = 0x16;      // Assign SDA1 output to RC1
    SSP1DATPPS = 0x11;      // Assign RC1         to SDA1 input
#endif
    SSP1CON1 = 0b00100110;  // SSPEN enabled, WCOL no collision, SSPOV no overflow,
                            // CKP low hold, SSPM I2C slave 7-bit

    SSP1CON2 = 0b00000000;  // ACKSTAT received, RCEN disabled, RSEN disabled, 
                            // ACKEN disabled, ACKDT acknowledge, SEN disabled, 
                            // GCEN disabled, PEN disabled

    SSP1CON3 = 0b00000000;  // BOEN disabled, AHEN disabled, SBCDE disabled, 
                            // SDAHT 100 ns hold, ACKTIM ackseq, DHEN disabled, 
                            // PCIE disabled, SCIE disabled

    SSP1STAT = 0x00;
    SSP1BUF  = 0x00;
    SSP1MSK  = 0xff;
    SSP1ADD  = I2C_SLAVE_ADDR << 1;
    PIR3bits.SSP1IF = 0;    // Clear the SSP Interrupt flag
    PIE3bits.SSP1IE = 1;    // Enable SSP Interrupts
    INTCONbits.GIE = 1;     // Enable global interrupts
    INTCONbits.PEIE = 1;    // Enable peripheral interrupts
    /*
     * Embedded code never returns from main
     */
    for(;;)
    {

    }
}

PPS настроен правильно, но я не знаю, правильный ли код инициализации I2C.

...