Volt/Amper/Power/Capacitance meter

Vladimir N. Shilov 557df622f9 Мелкий фикс. 7 years ago
EWSTM8 a84533fec1 Настройка проекта IAR-а 8 years ago
StdPerphDrv 6128cd6685 VAPC: добавил скелет проекта. 9 years ago
inc b8db42d441 VAPC: Configure SPI, out to LED. 9 years ago
lib f0b891e08e Борьба за точность показаний. 7 years ago
misc b8db42d441 VAPC: Configure SPI, out to LED. 9 years ago
sch ee272fd6f4 Первичная цель достигнута. Переходим ко второму этапу. 8 years ago
src 557df622f9 Мелкий фикс. 7 years ago
.gitignore be241e1ea4 Fix ignore list 8 years ago
Makefile 1153666a01 Подправил под новые условия. 8 years ago
ReadMe.txt 557df622f9 Мелкий фикс. 7 years ago
VAPC-meter.cbp e828a010e7 Ностройки проекта. 8 years ago

ReadMe.txt

** Проект VAPC-meth **
* Измеритель напряжения/тока/мощности/ёмкости *

Немогу никак определиться с диапазонами.
Думал сначал сделать по два диапозона на напряжение и ток, но что-то мне эта
идея разонравилась. Опять же, измерять большие напряжения/токи не особо часто
приходится. Напряжения в основном до 20 вольт, токи - до пары ампер.
Тем более, так как это измеритель в конкретное устройство, то нужно под него и
подстраиваться.
Будет по одному диапазону на ток и напряжения.

нужно сделать пару кнопок.
сделать ввод Кд делителя напряжения умноженного на 1000,
и сопротивление шунта * Ку услителя тока * 1000.

Далее в программе при вычислениях - получили код ацп, перевели в напряжение,
умножили на 1000 и поделили на коэффициент.

Ох и здоровая (85х40мм) плата получилась...
можно попробовать разбить на две -- на одной индикатор и кнопки, на второй всё
остальное. получится две платы ~ 45х40 мм.

вариант A -- поменял входной разъём.
вариант B -- поменял компановку, плата стала 51х47 мм.
вариант C -- убрал кварц, добавил двухцветный светодиод, плата стала 49.53 x 46.67 мм.
думаю последний вариант вполне неплох -- мне даже нравится :-)
можно использовать в качестве окончательного.
только вот нормы 0.4/0.17 -- можно пробовать фоторезистом.

по софту пока идей нету :-)

SVN-инит 2014.12.29

появилась мысль -- сделать транзисторный ОК выход, например для управления
релюхой. хотя если брать ардуиновское реле -- то там нужен просто выход.
реле для управления нагрузкой -- можно сделать тот-же зарядно-разрядный
автомат.
===
2014.12.30
вывел ещё два вывода на колодку.

---
2015.05.14

добавил скелет проекта, makefile, C::B проект.

---
2015.05.15

настройка SPI.
должно вывести 0-7 на индикаторы.

---
2015.05.18

добавил RTOS, перенёс туда timing_delay, задержка на 4-ом таймере 1 мс,
обработчик таймера перенёс в RTOS.

должно вывести 0-7, через 5 сек 8-f, и так по кругу.

---
2016.03.23

Проект уже на git.

Закончил паять (начал вчера), прошил.
И тишина...

Попутно исправил в схеме некоторые номиналы.
Шунт на 0.1 -- не влезет по мощности, использовал 0.05 Ом.

На плате есть мелкие огрехи, но самый конкретный косяк -- не те индикаторы.
По факту использовал корпуса 3-х символьных 7.62 мм. Втулил свои 4-х символьне
9 мм. подогнув выводы, но кнопки и светодиод не влезли.

Исправил дефайны индикаторов.

Осцилом ничего не увидел на выводах SPI, нужно искать.

Почитал даташит на тему SPI, поправил свой код.

---
2016.03.24

Так-как по прежнему тишь и глухомань, поднял проект в IAR-е и полез в
отладку. Всё застряло где-то в Delay - похоже на проблемы с таймером.
Таки да, забыл включить тактирование таймера -- похоже потерялось при
переносе инициализации то туда, то сюда...

Повторяем заповедь -- "Часы, ноги, перефирия".

Вобщем, SPI работает. Проблема была в том, что токозадающий резистор для
MAX нужно подключать на +5В, а у меня он был на земле. Напаял навесом.
Картинка появилась, но на индикаторах какой-то бред.

Исправил коды символов. Исправил позиции индикаторов.
Вылезло два момента -- неправильно запаял индикаторы, собирался красный
вниз, под ток, а жёлтый вверх под напряжение. В итоге запаял наоборот. Так
ещё и на схеме они местами попутаны, так как на плате DS1 внизу, а DS2
вверху. Исправил схему и позиции индикаторов под факт.

Добился картинки. Похоже тут какой-то бешенный оптимизатор -- выкинул нах
похожую функцию. Нужно учесть на будущее.

---
Пора браться за ADC.

С учётом входного сопротивления канала напряжения в 100 кОм,
на частоте 16 МГц нужно использовать 384 цикла измерений,
а на частоте 8 МГц -- 192 цикла.

Формула расчёта (AN2834 стр. 25) = (циклы / (f_adc * C_adc * ln(2^N+1)) - 1 кОм
ln(2^13) = 9,01091334728
C_adc = 16 * 10^-12, т.е.:
циклы/(f_MHz * 0.00016 * 9,01091334728) - 1 == кОм

С канала тока, по идее, можно мерять на любой скорости.

---
2016.03.25

Factory_VREFINT у меня == 0x7E, т.е. полное значение 0x067E == 1662
Если это мерялось при 3В, то получается -- 1.218 В

Предварительный успех -- читаю три канала АЦП и ПДП складывает их в
память, преобразовываю и вывожу на индикаторы. Запуск АЦП каждые 10 мсек
тригером от Т2. Настройка Т2 пока заклинанием -- понимания нету.
На очереди -- усреднение результатов, вычисление напряжения питания через
опорное напряжение.

Пока не сильно нравится -- завышает напругу, скачут показания.
Мерял аккум - 4.219 В, мой точный показывал 4.207, DT-шка - 4.17-4.18 В.
Канал внутреннего опорного напряжения показывает 1.219 - 1.223 В
По каналу тока висит 4 мВ, мой 5-ти разрядный вольтметр показывал 3 мВ на
выводе.
По каналу напряжения постоянно висит 17-26-35 мВ, вольтметр показывет
чёткий "0". Замыкание на землю ничего не даёт. Хз, что это такое.
Если от итогового напряжения вычесть эти миливольты, то результат будет
больше похож на правду.

На тему усреднения, такая идея:
- в прерывании ПДП данные перекладываются в быстрый кольцевой буфер
на 10 значений. Можно мерять чаще, хоть каждую 1 мс.
- в основном теле каждые 100 мсек отрабатывает процедура, которая
усредняет данные из быстрого буфера, отбрасывая слишком мелкие/большие,
и перекладывает в медленный кольцевой буфер. Тут же делать пересчёты
значений из кода АЦП в реальные значения.
- ещё одна процедура раз в секунду усредняет данные из медленного буфера
и сохраняем в текущий секундный результат. Здесь же вычеслять мощности,
ёмкости, время и т.д.

---
2016.03.25

Причесал настройку Т2.

Считаем среднее, зануляем младшие 4 бита, и потом сравниваем каждое
значение со сброшенными 4-мя младшими битами:
middle &= 0xFFF0;
if ((value & 0xFFF0) == middle) { ... }
таким образом отберём только те значения, которые отличаются от
предварительного среднего не более чем на 1/16

---
2016.03.28

Период измерений 1.25 мсек, быстрый буфер - на 16 значений.
Фильтра нет, только усреднение.
Понял, почему скачет опора -- при включеном светодиоде проседает
напряжение питания и завышаются показания. Что за хрень в канале напруги
всё равно не понятно.
Нужно срочно увеличивать резисторы в цепи светодиодов.
Без светодиодов канал внутренней опоры показывает 1.219 В.

---
2016.03.31

Начитался AN3137. Краткие выводы:
- конденсаторы по входу АЦП нужно поменять на 220 нФ,
- период измерения сделать не менее 2 мсек по упрощённой формуле,
или не менее 1,2 мсек по полной формуле,
- всё остальное -- отталкиваясь от этих данных.

Подробнее:

1. внешний конденсатор (правда в некоторых слачаях его не нужно ставить,
только я не понял в каких) расчитывается по так:
Cext ~= 1.58 * Csh * (Umax / Ulsb),
где Csh -- внутрення ёмкость АЦП, 16 пФ,
Umax -- максимальное входное напряжение, принимается 4096,
Ulsb -- нужная точность, принимается 0.5 LSB.
Итого, Cext ~= 1.58 * 0.000000000016 * (4096 / 0.5) >= 207 nF.
Этот конденсатор должен держать входное напряжение, пока оно меряеться АЦП,
при этом он должен разрядиться не больше чем на Ulsb.
Интересно, а можно его использовать для оверсемплинга, расчитав на 1 LSB ?
Конденсатор на 10 нФ успевает разрдится на 6-10 LSB.

2. Период времени между измерениями, за который должен успеть зарядиться
входной конденсатор, расчитывается так:
tc = - (Rin * Csh) * ln(1 - (Csh/Cext) * (Umax/Ulsb))
или, упрощённая формула tc ~= (Rin * Csh)
где Rin -- это внешнее сопротивление по отношению к АЦП (Rext)
ln(1 - (Csh/Cext) * (Umax/Ulsb)) ~= 0.9058
и в итоге, для канала вольтметра (как более высокоомного и медленного):
tc = (Rin * Csh) * 0.9058 = 9090 * 0.000000000016 * 0.9058 ~= 1,2 мсек

---
2016.07.27

Попытка "ADC2" провалилась -- там что-то не то в буфере от АЦП.
Возвращаюсь на последнюю рабочую версию и пойду другим путём.

План:
- меряем два канала (напряжение и ток) каждые 1,5625 мсек;
- DMA забирает данные, в прерывании перебрасываем два слова в первичный
циклический буфер на 64 значения (два буфера под напряжение и ток?)
в этом буфере сырые данные (код от ацп);
- раз в 100 мсек усредняем/обрабатываем эти 64 значения и складываем во
второй буфер на 10 значений;
- раз в секунду усредняем и считаем данные из второго буфера.

Закончил.
Получился слегка затянутым -- "фронт" изменения показаний около 3-х секунд.
Шаг вольтметра в 1 мВ я получил, но смысла особого не вижу :-)
Без "оверсэмплинга" показания более стабильны.

Фильтрация по каналу тока -- её не заметно. хз куда смотреть.
В канале тока постоянно "висит" 3-4 мА, когда-то я считал,
это похоже на смещение "0" от ОУ. Хз, нужно ли вычитывать их из результата.

Этот этап можно считать законченым.

---
Убрал оверсэмплинг по напряжению, и фильтрацию по току.

Идея по индикации:
- разбить на режимы:
* режим БП
+ вверху напряжение
+ внизу ток или мощность
* режим заряд/разряд
+ вверху напряжение / время (по кнопке или таймеру)
время мм.сс или чч.мм, тогда мигаем точкой синхронно с секундами.
+ внизу ток (заряд) / А*Ч (разряд) / Вт*Ч (по кнопке или таймеру)

Если две кнопки -- то двумя выбирать режим, светодиодом можно отображать.
Одиночое нажатие меняет выводимую инфу.
Индикация выбранного режима (секунд 5, можно с морганием): "PSU", "CHAr", "DECH";
текущего параметра: "-U-", "-I-", "-P-", "-t-", "-IH-", "-PH-"

Или проще -- выбираем, что отображать в текущей строке:
"-U-", "-I-", "-U-I-", "-P-", "-t-", "-IH-", "-PH-", "-U-t-", "-I-t-"
В двойных режимах менять отображаемый параметр по таймеру.

Обработка кнопок.
Каждая кнопка коротким нажатием переключает отображаемую величину своей строки.
Что повесить на длинное нажатие -- не могу придумать.
Как я понимаю, на каждую строку нужен свой КА, состояния которого будет
переключать соотвествующая кнопка.
Получается, что для каждой строки нужна функция отображения каждого параметра.
Как-то это накладно, но ничего умнее пока придумать не могу.
хватило бы оперативки...

---
2016.07.28

Сделал переключение режимов построчно.
Пора подумать о сохранении настроек в eeprom.

---
2016.12.01

В режимах отображения ёмкостей, времени длинным нажатие будем сбрасывать
счётчики ёмкостей и времени.
При старте, при обоих нажатых кнопках делать коррекцию 0 амперметра.

Наверное, нужно не каждые 100 мсек брать данные от АЦП, а в модуле АЦП
генерить флаг готовности данных. И объеденить быстрые буферы в одну структуру.

Нужно попробовать чтобы ПДП перебрасывал 2 по 2 по 64 байт, сразу в быстрый буфер.

Сделал чтобы напряжение и ток показывало не среднее за секунду, а последнее за 100 мсек.
Если будет сильно быстро, можно будет брать последние 2-4 результата.

---
2016.12.02

Что-то я вообще забыл, для чего я это всё начинал.
Если у меня получится переделать FSP235 в БП 30В 10А, то можно будет из этого
*-метра сделать что-то более полезное. Но для этого нужно прикрутить энкодер и
два ШИМ-выхода для управления БП. Плюс ещё рэле, или два. Попробуем.
- есть два выхода Т2 -- 1 и 2. на них однозначно шим.
- есть два входа АЦП, если смогу к ним припаяться -- туда можно прицепить
переменные резисторы для регулировки напряжения и тока. это будет удобнее и
проще, чем через энкодер.
- реле можно вместо светодиодов, но сначала придумать зачем они нужны.
- энкодер, программно, можно на ПА2, ПА3 а кнопку на ПД0.

По регулировке
- если выбрать шаг регулирования в 10 мВ и 10 мА, то для тока нужно 10 бит, а
для напряжения -- 12 бит.
- если ставить энкодер, то переменники не нужны.

По измерениям
- для экономии ОЗУ, можно в прерывании от ПДП суммировать в 32-бит
аккумуляторы, а в конце цикла доводить до 16-бит и скидывать в медленный
буфер.

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

---
2016.12.09

Итак, БП почти работает.
Входной делитель по напряжению -- оставляем 10 и 1 кОм 0.1% -- 36.3В/8.9мВ
Амперметр -- шунт с БП 0.01 Ом, при 10А даст 0.1В, в усилителе шунта меняем
резисторы 20кОм 0.1% на 33кОм 1%, получаем Ку=33. Если не менять -- получим
максимальный предел по току в 16.5А, шаг измерения ~4 мА -- пока можно и
оставить. С резистором 38.3 кОм получим ток 8.6А с шагом в ~2 мА, с резистором
на 30 кОм -- 11А и ~2.7 мА.

Добавил простенькую фильтрацию сразу после DMA.
Вернул "оверсемплинг". Не уверен, правильно ли я насчитал коэфф...
Нужно будет добавить коррекцию "0" с выхода ОУ.

---
2017.06.05

Напряжение показывает нормально, ток -- задержка в пару секунд.
Или там другая прошивка, или я не знаю...
Чуть упростил вычисления напряжения на шунте -- я уже вообще не помню, что
там и как...
БП у меня может выдавать до 16 А, а настроен вроде на 6.6А...

---
2017.06.06

Рассматривая плату...
Дорожки обрывал...

Делитель по напряжению -- 100 и 10 кОм. (Кд = 11)

Усилитель шунта - 1 и 10 кОм, Ку = 10.
Шунт на БП = 0.05 Ом, максимальный ток - 6.6 А.
Если не менять шунт (а этого не хочеться), то для тока
16А -- Ку=4.125, шаг ~4мА
12А -- Ку=5.5, шаг ~3мА
10А -- Ку=6.6, шаг ~2.4мА
8А -- Ку=8.25, шаг ~2мА

Есть такие резисторы 1%:
Ку мах А шаг ~ мА
3.9К 3.9 16.923 4.1
5.62К 5.62 11.744 2.9
6.8К 6.8 9.706 2.4
9.1К 9.1 7.253 1.8

Если 1К заменить на 2К 0.5%, то:
Ку мах А шаг
11К 5.5 12.00 2.9
12К 6 11.00 2.7
15К 7.5 8.8 2.1
18К 9 7.333 1.8

Пока больше склоняюсь к варианту 2/11 - 12А.

И нужно будет увешать всё фильтрующими дроселями.

---
2017.06.22

Вместо 10 кОм поставил пару резисторов по 5.624 кОм.

Итоговй Ку мерял своим UT61E.
Также по току пришлось ввести поправочный коэфф., иначе показания заметно
отличались от UT61E.

По напряжению -- вполне нормально.

Осталось в самом БП поправить резисторы под 12А макс. тока и пора
корпусировать.

Так и не понял почему тормозили показания. Нужно попытаться разобраться.

---
2017.06.26

Понял -- я брал индекс следующего элемента в массиве, т.е. того, который
был секунду назад. Переделал.

Думал сделать гашение незначащих нулей, но забил.

Плату развёл дебильно -- теперь думай как её присобачить к корпусу...