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

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


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

Реклама: ООО ТД Промэлектроника, ИНН: 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 пользователей онлайн

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

  • Сообщения

    • Шота дохрена. Это если мультик может на килогерце. Обычно нет, предел 400 Гц.
    • Да, сначала был написан бутлодер для тини 24. Там нет выделенной памяти под бутлодер и заморочек с областями откуда можно записывать и перезаписывать флеш из программы, там все просто! Также там нет УАРТА, пришлось написать. Ну и самое главное: перезаписывай любые страницы флеш из любого места! В тини 24 под бутлодер ушло 48% памяти. В проекте технологи выделили слишком мало физического места под схему. Потом развел плату, оказалось, что места не так уж и мало. Потом выяснилось, что пинов впритык, и если будет дальнейшее расширение, их может не хватить. И после долгих (ну очень долгих) изысканий (цена, доступность, размеры) остановился на 88 (очень низкая цена, корпус чуть больше чем у 24, датчик температуры, а это важно, но не критически). Да, и сейчас все программируют на С и др. языках высокого уровня, что занимает очень (ну очень) много места, и чипы с небольшой памятью остаются не востребованы, и как следствие низкий спрос, и низкая цена!!! И при тупом переносе программы перезаписи страниц флеш программа не заработала... Ну и пошли изыскания, и я набрел на ВАШ проект, самый полный и законченный на то время. Нужно было выяснить чип который был у меня битый или я что-то упустил при программировании? (в бутлодерах я новичек) Недостаток 88 на сегодня, то что из пользовательской памяти нельзя перепрограммировать флеш, хотя страница стирается.   По поводу ВАШЕЙ программы: выяснилось, что при включенном фьюзе WDT в конфигурации, соединение не устанавливается, а в хелпе это не отражено. В некоторых ситуациях соединение можно восстановить лишь отключив / включив питание, сброс через ножку ресет не помагает. Будем копать дальше,... и глубже....!   Иногда при "установить соединение" выдается сообщение, "соединение не установлено", а на осциллографе виден ответ с чипа?
    • Сейчас по входу стоит керамика 0,1 мкФ + электролит 220 мкФ, иголок вроде нет, ну во всяком случае старичек С1-93 их не показывает.
    • Платы материнки и УВ пока в брак.  - Семён Семёныч... (с) Вы ничего не забыли, сэр? Как вы собрались регулировать УВ по уровню выходного сигнала??? Решение уже есть, пока в процессе. Вот так оно лучше работает.
    • Ну в идеале каптоновым скотчем как вам сказали,но можно обмотать чем угодно лижбе диалектриком было и хотябы немного высокотемпературное,я бы термо усадку взял,поидеи хорошо пойдет  Ещё видел где трансформатор медно алюминиевой жилой был на ампер 10 если не больше на вид аж,и там алюминиевая обмотка сечением пол сантиметра была замотана в бумагу пропитанную в воске,слоем миллиметр примерно эта изоляция была 
    • Всем привет. Собрал мх50, запустился с первого раза, когда делал замеры, соскочил щуп, спалил выходники и пред, заменил, все ок. Питание тор 250 ватт 30х2 выход, после выпрямителя 44В 20000мкф в плечо  увеличил входные емкости на оконечниках, добавил шунтирующую пленку  плата софт старта присутсвует цепь буше взял из ОМ2.7 ТП выставил 100мА добавлена термозащита на каждый канал( но как оказалось, с таким корпусом хрен перегреешь) АС kef q150 dac Aiyima a5 max получилось снять 81 ватт с канала при 8ом нагрузке (на фото самое точное измерение на левом мультиметре) без сигнала если прислонится к ас еле уловимое шипение  при наличии сигнала все ок, играет отлично, мне очень понравилось. в планах ОМ2.7
    • Уже 52 скачивания. Значит актуальна.
  • Похожий контент

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