реклама на сайте
подробности

 
 
 
Reply to this topicStart new topic
> Запись флеш в LPC1778, Ответвление от темы про стартовый загрузчик
ДЕЙЛ
сообщение Oct 2 2014, 09:53
Сообщение #1


Местный
***

Группа: Участник
Сообщений: 234
Регистрация: 7-11-13
Пользователь №: 79 085



Решил данный вопрос задать в отдельной теме, ответвив от темы про стартовый загрузчик http://electronix.ru/forum/index.php?showt...15&start=15
Там подсказали насчёт организации записи флеш из пользовательской программы. Сделал по мануалу, т.к. примеров пока не видел. Получилось вот так:
Код
#include "iolpc1778.h"
#include "LPC17xx.h"
#define IAP_LOCATION 0x1fff1ff1 //точка входа в IAP (страница 896 мануала)  
unsigned long *command;
unsigned long *result;
unsigned int  *test;
typedef void (*IAP)(unsigned int*, unsigned int*);  
void main(void)
{
{ //INIT
SCS     |= 0x20;          //подключение осциллятора 12МГц

while(!( SCS & 0x40 )) {} //ожидание запуска

PLL0CON |= 0x01;          //включение PLL0
PLL0CFG |= 0x09;          //умножение частоты тактирования на (1+9) - 120МГц
PLL0FEED = 0xAA;          //
PLL0FEED = 0x55;          //
CCLKSEL |= 0x100;         //тактирование CPU от PLL0
}
test  = (unsigned int*)0x20001004; //записал контрольный байт в копируемую область RAM
*test = 0xAA;



IAP iap_entry;                              //так в мануале написано
iap_entry = (IAP) IAP_LOCATION; //---//---



//_______СТЕРЕТЬ СЕКТОР_______//
{
result  = (unsigned long*)0x20006100;   //адрес расположения таблицы результатов
command = (unsigned long*)0x20006000;   //адрес таблицы параметров

*command = 52; //код команды
*command++;
*command = 4;  //начальный номер сектора
*command++;
*command = 4;  //конечный номер сектора
*command++;
*command = 120000; //системная тактовая частота в кГц

asm ("mov  r0, #0x6000"); //Запись в регистры отдельно младшие 2 байта
asm ("movt r0, #0x2000"); //и старшие два байта адресов таблиц

asm ("mov  r1, #0x6100");
asm ("movt r1, #0x2000");

iap_entry (command, result);
}
//_______СТЁРЛИ СЕКТОР______//





//_______ПОДГОТОВИТЬ СЕКТОР К ЗАПИСИ_______//
{
result  = (unsigned long*)0x20006100;   //адрес расположения таблицы результатов
command = (unsigned long*)0x20006000;   //адрес таблицы параметров

*command = 50; //код команды
*command++;
*command = 4;  //начальный номер сектора
*command++;
*command = 4;  //конечный номер сектора

asm ("mov  r0, #0x6000"); //Запись в регистры отдельно младшие 2 байта
asm ("movt r0, #0x2000"); //и старшие два байта адресов таблиц

asm ("mov  r1, #0x6100");
asm ("movt r1, #0x2000");

iap_entry (command, result);
}
//_______ПОДГОТОВИЛИ СЕКТОР К ЗАПИСИ______//







//_______КОПИРОВАТЬ ОПЕРАТИВНУЮ ПАМЯТЬ ВО ФЛЕШ_______//
{
result  = (unsigned long*)0x20006100;   //адрес расположения таблицы результатов
command = (unsigned long*)0x20006000;   //адрес таблицы параметров

*command = 51;         //код команды
*command++;
*command = 0x4000;     //начальный адрес перезаписываемой флеш
*command++;
*command = 0x20001000; //начальный адрес оперативной памяти, откуда нужно копировать

*command++;
*command = 512;        //число байт для копирования
*command++;
*command = 120000;     //системная тактовая частота в кГц


asm ("mov  r0, #0x6000"); //Запись в регистры отдельно младшие 2 байта
asm ("movt r0, #0x2000"); //и старшие два байта адресов таблиц

asm ("mov  r1, #0x6100");
asm ("movt r1, #0x2000");

iap_entry (command, result);
}
//_______СКОПИРОВАЛИ СЕКТОР______//




while(1)
{
   test  = (unsigned int*)0x4004; //для просмотра ячейки в отладчике в пошаговом режиме.
}
}



Работать почему-то не хочет - как было во флеш FFFFFFFFFFFF, так и остались. Ещё в мануале написано, что содержимое регистров r0 и r1 остаётся постоянным во время работы с флеш-памятью, но у меня в пошаговом режиме видно, что после функции iap_entry (command, result); они меняются. Так же там написано, что в регистры r0 и r1 записываются начальные адреса таблиц параметров и результатов. Можно ли в моём коде избавиться от ассемблерных вставок, т.к. каждый раз вручную назначать адреса таблиц в памяти неудобно и нежелательно. Ещё там написано, что верхние 32 байта RAM не должны использоваться пользовательской программой, но как программе запретить пользоваться этими байтами?

UP1: добавил файл проекта в IAR

Сообщение отредактировал ДЕЙЛ - Oct 2 2014, 11:24
Прикрепленные файлы
Прикрепленный файл  20141001LPC.rar ( 899.54 килобайт ) Кол-во скачиваний: 5
 
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Oct 2 2014, 10:39
Сообщение #2


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(ДЕЙЛ @ Oct 2 2014, 12:53) *
Работать почему-то не хочет
Перед командой стирания надо выполнить команду подготовки сектора. Запись в регистры асм-вставками бессмысленна - вы делаете то же самое вызывая iap_entry() с параметрами. Во стальном похоже на правду. Вы зарезервировали для IAP место в конце ОЗУ?


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
ДЕЙЛ
сообщение Oct 2 2014, 11:19
Сообщение #3


Местный
***

Группа: Участник
Сообщений: 234
Регистрация: 7-11-13
Пользователь №: 79 085



Цитата(Сергей Борщ @ Oct 2 2014, 14:39) *
Перед командой стирания надо выполнить команду подготовки сектора. Запись в регистры асм-вставками бессмысленна - вы делаете то же самое вызывая iap_entry() с параметрами. Во стальном похоже на правду. Вы зарезервировали для IAP место в конце ОЗУ?

С этим тоже вопрос. Как зарезервировать место в памяти?
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Oct 2 2014, 11:27
Сообщение #4


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(ДЕЙЛ @ Oct 2 2014, 14:19) *
Как зарезервировать место в памяти?
Читать документацию на среду разработки. Или хотя бы указать ее название - возможно кто-то и подскажет. Но читать полезнее sm.gif


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
ДЕЙЛ
сообщение Oct 2 2014, 11:38
Сообщение #5


Местный
***

Группа: Участник
Сообщений: 234
Регистрация: 7-11-13
Пользователь №: 79 085



Цитата(Сергей Борщ @ Oct 2 2014, 15:27) *
Читать документацию на среду разработки. Или хотя бы указать ее название - возможно кто-то и подскажет. Но читать полезнее sm.gif

среда IAR

Цитата(Сергей Борщ @ Oct 2 2014, 14:39) *
Вы зарезервировали для IAP место в конце ОЗУ?

В мануале написано, что нужно зарезервировать верхние байты. Что имеется ввиду под верхними байтами, если считать за начало адрес 0x2000000, а за конец 0x2008000? Эти 32 байта распложены в области 0x2000000 - 0x200001F или в другом конце?
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Oct 2 2014, 13:37
Сообщение #6


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(ДЕЙЛ @ Oct 2 2014, 14:38) *
среда IAR
Читайте про линкер и его скрипт. Возможно можно где-то в менюшках уменьшить доступный размер ОЗУ. Просто скажите линкеру, что у вашего процессора ОЗУ меньше на 32 байта.
Цитата(ДЕЙЛ @ Oct 2 2014, 14:38) *
или в другом конце?
В другом - в самом конце ОЗУ.


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
XVR
сообщение Oct 3 2014, 09:47
Сообщение #7


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



Цитата(Сергей Борщ @ Oct 2 2014, 14:39) *
Запись в регистры асм-вставками бессмысленна - вы делаете то же самое вызывая iap_entry() с параметрами.
Не совсем. ТС модифицировал command перед тем, как передавать его в iap_entry(). Так что работать не будет rolleyes.gif
Go to the top of the page
 
+Quote Post
ДЕЙЛ
сообщение Oct 3 2014, 10:08
Сообщение #8


Местный
***

Группа: Участник
Сообщений: 234
Регистрация: 7-11-13
Пользователь №: 79 085



Спасибо, заработало :-)

Выкинул ассемблер, упростил и в итоге получился работающий код:
Код
#include "iolpc1778.h"
#include "LPC17xx.h"
#define IAP_LOCATION 0x1fff1ff1 //точка входа в IAP (страница 896 мануала)  
unsigned int command[5];
unsigned int result[2];
unsigned int  *test;
typedef void (*IAP)(unsigned int[], unsigned int[]);  
void main(void)
{
{ //INIT
SCS     |= 0x20;          //подключение осциллятора 12МГц

while(!( SCS & 0x40 )) {} //ожидание запуска

PLL0CON |= 0x01;          //включение PLL0
PLL0CFG |= 0x09;          //умножение частоты тактирования на (1+9) - 120МГц
PLL0FEED = 0xAA;          //
PLL0FEED = 0x55;          //
CCLKSEL |= 0x100;         //тактирование CPU от PLL0
}
test  = (unsigned int*)0x20001004;
*test = 0xAA;



IAP iap_entry;
iap_entry = (IAP) IAP_LOCATION;



//_______ПОДГОТОВИТЬ СЕКТОР К ЗАПИСИ_______//
{

command[0] = 50; //код команды
command[1] = 4;  //начальный номер сектора
command[2] = 4;  //конечный номер сектора

iap_entry (command, result);
}
//_______ПОДГОТОВИЛИ СЕКТОР К ЗАПИСИ______//




//_______СТЕРЕТЬ СЕКТОР_______//
{

command[0] = 52;     //код команды
command[1] = 4;      //начальный номер сектора
command[2] = 4;      //конечный номер сектора
command[3] = 120000; //системная тактовая частота в кГц

iap_entry (command, result);
}
//_______СТЁРЛИ СЕКТОР______//

//_______ПОДГОТОВИТЬ СЕКТОР К ЗАПИСИ_______//
{

command[0] = 50; //код команды
command[1] = 4;  //начальный номер сектора
command[2] = 4;  //конечный номер сектора

iap_entry (command, result);
}
//_______ПОДГОТОВИЛИ СЕКТОР К ЗАПИСИ______//




//_______КОПИРОВАТЬ ОПЕРАТИВНУЮ ПАМЯТЬ ВО ФЛЕШ_______//
{

command[0] = 51;         //код команды
command[1] = 0x4000;     //начальный адрес перезаписываемой флеш
command[2] = 0x20001000; //начальный адрес оперативной памяти, откуда нужно копировать
command[3] = 512;        //число байт для копирования
command[4] = 120000;     //системная тактовая частота в кГц
  
iap_entry (command, result);
}
//_______СКОПИРОВАЛИ СЕКТОР______//

while(1)
{
   test  = (unsigned int*)0x4004;
}
}


Интересно узнать, в чём суть подготовки памяти? Со стиранием вроде понятно - наверно FFFFFFFами заполняет всё. И ещё есть не совсем понятная команда заполнения пробелами. Правильно я понимаю, что 0x1fff1ff1 есть адрес начала отдельной программы IAP?
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Oct 3 2014, 12:53
Сообщение #9


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(ДЕЙЛ @ Oct 3 2014, 13:08) *
Спасибо, заработало :-)
"И совсем не страшно" sm.gif

Цитата(ДЕЙЛ @ Oct 3 2014, 13:08) *
Интересно узнать, в чём суть подготовки памяти?
Это знают только авторы IAP.
Цитата(ДЕЙЛ @ Oct 3 2014, 13:08) *
Правильно я понимаю, что 0x1fff1ff1 есть адрес начала отдельной программы IAP?
Да.


Цитата(XVR @ Oct 3 2014, 12:47) *
Не совсем. ТС модифицировал command перед тем, как передавать его в iap_entry(). Так что работать не будет rolleyes.gif
Недопонял. Ну сделал он его указателем, как это повлияло на неработу? Адрес массива передается в функцию точно таким же указателем. Место в памяти под command он выделяет статически. Да, абсолютные адреса-"магические цифры", криво но как может влиять на неработу? Или я чего-то другого не заметил?


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
XVR
сообщение Oct 3 2014, 16:53
Сообщение #10


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



Цитата(Сергей Борщ @ Oct 3 2014, 16:53) *
Недопонял. Ну сделал он его указателем, как это повлияло на неработу? Адрес массива передается в функцию точно таким же указателем.

Он несколько раз инкрементировал этот указатель, прежде чем передать в функцию:
Код
command = (unsigned long*)0x20006000;   //адрес таблицы параметров

*command = 51;         //код команды
*command++;  // <<< тут
*command = 0x4000;     //начальный адрес перезаписываемой флеш
*command++; // <<< тут
*command = 0x20001000; //начальный адрес оперативной памяти, откуда нужно копировать

*command++; // <<< тут
*command = 512;        //число байт для копирования
*command++; // <<< и тут
*command = 120000;     //системная тактовая частота в кГц

iap_entry (command, result); // В результате тут command смотрит не на команды, а сразу ЗА НИМИ
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Oct 3 2014, 18:33
Сообщение #11


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Да, действительно.


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
ДЕЙЛ
сообщение Oct 6 2014, 18:14
Сообщение #12


Местный
***

Группа: Участник
Сообщений: 234
Регистрация: 7-11-13
Пользователь №: 79 085



У меня было ошибочное предположение, что адреса таблиц нужно вписывать в регистры вручную перед вызовом IAP, что я и делал:
Код
asm ("mov  r0, #0x6000"); //Запись в регистры отдельно младшие 2 байта
asm ("movt r0, #0x2000"); //и старшие два байта адресов таблиц

asm ("mov  r1, #0x6100");
asm ("movt r1, #0x2000");

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

Сообщение отредактировал ДЕЙЛ - Oct 6 2014, 18:15
Go to the top of the page
 
+Quote Post
XVR
сообщение Oct 7 2014, 07:53
Сообщение #13


Гуру
******

Группа: Свой
Сообщений: 3 123
Регистрация: 7-04-07
Из: Химки
Пользователь №: 26 847



Цитата(ДЕЙЛ @ Oct 6 2014, 22:14) *
Поэтому в моём представлении инкременты ничего не меняли, но в любом случае вариант кода из первого поста не работал.
Так по этому и не работал - передача параметрами command и result полностью забивали то, что вы сделали в ассемблерной вставке. Причем меняли правильные значения в регистрах (после ассемблерной вставки) на неправильные (из модифицированного command)

Go to the top of the page
 
+Quote Post
ДЕЙЛ
сообщение Oct 7 2014, 12:31
Сообщение #14


Местный
***

Группа: Участник
Сообщений: 234
Регистрация: 7-11-13
Пользователь №: 79 085



Цитата(XVR @ Oct 7 2014, 11:53) *
Так по этому и не работал - передача параметрами command и result полностью забивали то, что вы сделали в ассемблерной вставке. Причем меняли правильные значения в регистрах (после ассемблерной вставки) на неправильные (из модифицированного command)

не работало по причине неправильной программы, это я уже выяснил sm.gif Этот вопрос можно пока закрыть. Лучше посоветуйте в теме про стартовый загрузчик.
Go to the top of the page
 
+Quote Post

Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 23rd July 2025 - 23:11
Рейтинг@Mail.ru


Страница сгенерированна за 0.01664 секунд с 7
ELECTRONIX ©2004-2016