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

Вопросы от начинающих по МК


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

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

  • Ответов
  • Создана
  • Последний ответ

Топ авторов темы

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

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

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

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

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

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

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

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

вновь нужна ваша помощь друзья. 

в примере на мой мк меняют регистры следующим образом

void PSC_Load (unsigned short dt0a, unsigned short dt1a, 
unsigned short dt0b, unsigned short dt1b,
unsigned short dt0c, unsigned short dt1c)
{
	PCNF0 = (1 << POP0) | (1 << PLOCK0) | (1 << PMODE01) | (1 << PMODE00) | (1 << PCLKSEL0);
	PCNF1 = (1 << POP1) | (1 << PLOCK1) | (1 << PMODE11) | (1 << PMODE10) | (1 << PCLKSEL1);
	PCNF2 = (1 << POP2) | (1 << PLOCK2) | (1 << PMODE21) | (1 << PMODE20) | (1 << PCLKSEL2);

	OCR0SA = dt0a;
	OCR0SB = dt1a;
	OCR1SA = dt0b;
	OCR1SB = dt1b;
	OCR2SA = dt0c;
	OCR2SB = dt1c;

	PCNF0 = (1 << POP0) | (1 << PMODE01) | (1 << PMODE00) | (1 << PCLKSEL0);
	PCNF1 = (1 << POP1) | (1 << PMODE11) | (1 << PMODE10) | (1 << PCLKSEL1);
	PCNF2 = (1 << POP2) | (1 << PMODE21) | (1 << PMODE20) | (1 << PCLKSEL2);
}

почему не писать проще, PCNF0 |=(1<<PLOCK0);  и закрывать PCNF0 &=~(1<<PLOCK0); а все остальное оставить в init

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

 if (inst_volt>volt)
		 {
			 OCR2RA++;
			 OCR2RB++;
		 }
		 if (inst_volt<volt)
		 {
			 OCR2RA--;
			 OCR2RB--;
		 }

вот так работает. и PLOCK бит совсем не нужен. почему то эти регистры не пишутся вообще если их объявить через переменную, типа unsignet int On_time1 = OCR2RA;  и затем уже On_time1++;

поэтому придется писать чтобы не падали ниже нуля. и еще частота с которой эти регистры изменяются очень маленькая

и еще такой вопрос будет ли корректно писать так:

if (inst_volt=volt) 
{ 
	OCR2RA=OCR2RA;
	OCR2RB=OCR2RB;
}

 

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

58 minutes ago, Дмитрий Вас said:

будет ли корректно

Ну, при сравнении if() надо использовать двойной знак равенства "==".

В остальном - корректно, но абсолютно бесполезно. Если inst_volt равно volt, то просто ничего не надо делать, и регистры OCR2RA и OCR2RВ просто не изменятся.

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

Здравствуйте. Попала в руки горсть силабов c8051f350 с хорошим ацп. Программатор к ним тоже есть, но совсем нет примеров работ на Си.

Может направите в нужное русло, литературы не нашёл. На асамблере инфы по 8051 полно, а на си практически нет.

Даже элементарного примера работы с портами нет (например помогать светодиодом не удалось найти..)

Если у кого есть проекта на си 8051-скиньте пжлст, хоть будет примером.

 

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

3 минуты назад, _abk_ сказал:

 

Видимо да)

Спасибо, буду грызть.

Странно, что для такого массового контроллера так мало информации по сравнению с АВР и пик

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

Читаю даташит на тринадцатую тиньку. Возник вопрос по адресному пространству. Есть регистры общего назначения, есть регистры ввода-вывода. У них разные имена, но есть пересечения по адресам. Например, адрес 0x1E это и EEPROM data register в регистрах ввода-вывода, и R30 в регистрах общего назначения, то есть в оперативной памяти. Правильно ли я понимаю, что физически это разные адреса, и для работы с ними используются разные наборы команд?

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

  • 2 недели спустя...

Уважаемые участники сообщества, прошу помощи. Задача у меня такая: Хочу просто сделать таймер обратного отсчета. Взял мк авр, 7 сегментный индикатор поставил авр студию, протеус, пишу на си. По моей задаче, собственно запрограммировал 0 таймер на прерывание и отображение информации на индикаторе. Всё работает, цифры показывает, используя  _delay_ms() таймер тикает  всё работает. Но все знают что _delay_ms() это бичёвская команда. И как от неё избавится. Предложение функция millis от ардуино. Но вот вопрос. как её реализовать на си в авр студии?

Я попытался инкрементировать переменную в обработчике прерывания. После чего в основном цикле значение этой переменной присваиваю другой переменной и дальше с ней работаю. Следующим образом. Когда переменная станет равна 1000, я хочу увеличить значение на индикаторе на единицу. Итог нехрена не работает как я хочу. Код прилагаю. Объясните где и что я делаю не так. 

#define F_CPU 8000000
#include <avr/io.h>
#include <avr/interrupt.h>
#include <stdbool.h>
#include <util/delay.h>

//Инициализация глобальных переменных
int8_t n_count=0, caunterTimer0=1, razrad1=0, razrad2=0;
int16_t timerState=0,timerCaunter=0;

void Na_ekran(int8_t razbivka_chisla); //прототип функции разбивки числа на десятки и единицы
void segment(uint8_t seg);				//Прототип функции выбора сигмента


int main(void)
{
	//Настройка портов управления 7сегментного индикатора с общим анодом (возможно катодом не помню)
	DDRB = 0b11111111; //Настраиваем порт "В" на выход для работы с 7ми сегментным индикатором
	PORTB = 0b00000000;//
	DDRD|=(1<<PD0)|(1<<PD2)|(1<<PD3)|(1<<PD5)|(1<<PD6);
	PORTD|=(1<<PD2)|(1<<PD3);
	
	//Настройка теймера0
	TIMSK|=(1<<TOIE0);//Активация режима разрешение прерывания по переполнению для таймера счётчика0
	TCCR0B|=(0<<CS02)|(1<<CS01)|(0<<CS00);// делим частоту таймера на 8 
	TCNT0=0;//Обнуление счётного резистра таймера счётчика0
	
	int16_t caunter=0;

	while (1)
	{
		cli();
		timerCaunter=timerState;
		sei();

		if (timerCaunter==1000)
		{
			Na_ekran(caunter+=1);
		}
		cli();
		if (timerCaunter>1000)
		{
			timerState=0;
		}
		sei();
	}
}


ISR (TIMER0_OVF_vect)
{
	
	if(caunterTimer0==0)
	{
		PORTD&=~(1<<PD5);
		PORTD|=(1<<PD6);
		segment(razrad1);
	}
	if(caunterTimer0==1)
	{
		PORTD&=~(1<<PD6);
		PORTD|=(1<<PD5);
		segment(razrad2);
	}
	caunterTimer0++;
	
	
	if (caunterTimer0>1)
	{
		caunterTimer0=0;
	}
	timerState++;
	if (timerState>1000)
	{
		timerState=0;
	}
	
	
	
}
void Na_ekran (int8_t razbivka_chisla) //функция разбивки числа на десятки и единицы
{
	razrad1= razbivka_chisla/10;//десятки
	razrad2= razbivka_chisla%10;//еденицы
	
}
void segment (uint8_t seg) //функция воспроизведения цыфры на 7-сегментном индикаторе
{
	switch(seg)
	{
		case 1: PORTB = 0b11111001; break;
		case 2: PORTB = 0b10100100; break;
		case 3: PORTB = 0b10110000; break;
		case 4: PORTB = 0b10011001; break;
		case 5: PORTB = 0b10010010; break;
		case 6: PORTB = 0b10000010; break;
		case 7: PORTB = 0b11111000; break;
		case 8: PORTB = 0b10000000; break;
		case 9: PORTB = 0b10010000; break;
		case 0: PORTB = 0b11000000; break;
	}
}

 

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

caunterTimer0 равен нулю при первом прерывании и включает первый разряд, а потом инкрементирует эту  переменную. При следующим прерывании, включает второй разряд. При третьем при caunterTimer0 равен уже трем и должен сброситься в ноль.

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

@azgo  Надо объявить переменную timerState с волшебным словом volatile.

volatile int16_t timerState=0

....

int main(void)
{
....
	while (1)
	{
		if (timerState >= 1000)
		{
			caunter += 1			
			Na_ekran(caunter);
			timerState = 0;
		}
	}
}

А из обработчика прерывания зануление timerState=0; убрать.

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

Как то так

Спойлер
#define F_CPU 8000000
#include <avr/io.h>
#include <avr/interrupt.h>
#include <stdbool.h>
#include <util/delay.h>

//Инициализация глобальных переменных
int8_t n_count=0, caunterTimer0=1, razrad1=0, razrad2=0;
int16_t timerCaunter=0;

volatile int16_t timerState=0;

void Na_ekran(int8_t razbivka_chisla); //прототип функции разбивки числа на десятки и единицы
void segment(uint8_t seg);				//Прототип функции выбора сигмента

int main(void)
{
	//Настройка портов управления 7сегментного индикатора с общим анодом (возможно катодом не помню)
	DDRB = 0b11111111; //Настраиваем порт "В" на выход для работы с 7ми сегментным индикатором
	PORTB = 0b00000000;//
	DDRD |= (1<<PD0)|(1<<PD2)|(1<<PD3)|(1<<PD5)|(1<<PD6);
	PORTD |= (1<<PD2)|(1<<PD3);
	
	//Настройка теймера0
	TIMSK |= (1<<TOIE0);//Активация режима разрешение прерывания по переполнению для таймера счётчика0
	TCCR0 |= (0<<CS02)|(1<<CS01)|(0<<CS00);// делим частоту таймера на 8
	TCNT0=0;//Обнуление счётного резистра таймера счётчика0
	
	int8_t caunter=0;
	sei();

	while (1)
	{

		if(timerState){ // ~257ms
			cli();
			Na_ekran(caunter);
			caunter++;
			if(caunter > 99){caunter = 0;}			
			timerState = 0;
			sei();	
		}

	}
}

ISR (TIMER0_OVF_vect) // 0.26ms
{
	if(caunterTimer0==0 && razrad1)
	{
		PORTD&=~(1<<PD5);
		PORTD|=(1<<PD6);
		segment(razrad1);
	}
	if(caunterTimer0==1)
	{
		PORTD&=~(1<<PD6);
		PORTD|=(1<<PD5);
		segment(razrad2);
	}
	caunterTimer0++;
	
	if (caunterTimer0 > 1){caunterTimer0 = 0;}
	
	if(timerCaunter) timerCaunter--;
	else{timerCaunter = 1000; timerState = 1;}

}


void Na_ekran (int8_t razbivka_chisla) //функция разбивки числа на десятки и единицы
{
	razrad1= razbivka_chisla/10;//десятки
	razrad2= razbivka_chisla%10;//еденицы
	
}
void segment (uint8_t seg) //функция воспроизведения цыфры на 7-сегментном индикаторе
{
	switch(seg)
	{
		case 1: PORTB = 0b11111001; break;
		case 2: PORTB = 0b10100100; break;
		case 3: PORTB = 0b10110000; break;
		case 4: PORTB = 0b10011001; break;
		case 5: PORTB = 0b10010010; break;
		case 6: PORTB = 0b10000010; break;
		case 7: PORTB = 0b11111000; break;
		case 8: PORTB = 0b10000000; break;
		case 9: PORTB = 0b10010000; break;
		case 0: PORTB = 0b11000000; break;
	}
}

 

И частота вывода на дисплей у вас высока. В Протеусе покажет, в железе едва.

Ой, регистр TCCR0B забыл под ваш МК вернуть :)

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

Спасибо уважаемые исправление кода. А можно теоретический комментарий, почему всё таки у меня не работало. Завтра попробую ваши изменения.

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

13 часов назад, Berg сказал:

Я и не вникал в ваш код, у меня из глаз кровь пошла от переменных: caunterTimer0, timerState, timerCaunter, caunter. :D

Ну за это мне уже напихали на другом форуме. Может есть методичка по клин коду? И что то для начинающих по програмированию авр контроллеров?

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

В 24.03.2022 в 09:38, Огонёк сказал:

Правильно ли я понимаю, что физически это разные адреса, и для работы с ними используются разные наборы команд?

Не совсем так.
Адресное пространство всего, что не ПЗУ - одно.

Обращение к ОЗУ по адресу 0 - это регистр R0

Обращение к ОЗУ по адресу 0x20 - это 0x00 адрес пространства "периферии"

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

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

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

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

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

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

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

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

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

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

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

×
×
  • Создать...