blob: 5f21d5c38e5b2c8eb7041c178ebe47e57420bb7e [file] [log] [blame]
Wolfgang Denkea882ba2010-06-20 23:33:59 +02001/*
2 * LowLevel function for DataFlash environment support
wdenk5779d8d2003-12-06 23:55:10 +00003 * Author : Gilles Gastaldi (Atmel)
4 *
Wolfgang Denk1a459662013-07-08 09:37:19 +02005 * SPDX-License-Identifier: GPL-2.0+
wdenk5779d8d2003-12-06 23:55:10 +00006 */
7#include <common.h>
wdenk5779d8d2003-12-06 23:55:10 +00008#include <command.h>
9#include <environment.h>
10#include <linux/stddef.h>
wdenk5779d8d2003-12-06 23:55:10 +000011#include <dataflash.h>
Wolfgang Denkea882ba2010-06-20 23:33:59 +020012#include <search.h>
13#include <errno.h>
wdenk5779d8d2003-12-06 23:55:10 +000014
Wolfgang Denkd87080b2006-03-31 18:32:53 +020015DECLARE_GLOBAL_DATA_PTR;
16
Igor Grinberg0901d9f2011-11-07 01:14:02 +000017env_t *env_ptr;
wdenk5779d8d2003-12-06 23:55:10 +000018
Igor Grinberg0901d9f2011-11-07 01:14:02 +000019char *env_name_spec = "dataflash";
Wolfgang Denkea882ba2010-06-20 23:33:59 +020020
Wolfgang Denkea882ba2010-06-20 23:33:59 +020021uchar env_get_char_spec(int index)
wdenk5779d8d2003-12-06 23:55:10 +000022{
23 uchar c;
Wolfgang Denkea882ba2010-06-20 23:33:59 +020024
Igor Grinberg0901d9f2011-11-07 01:14:02 +000025 read_dataflash(CONFIG_ENV_ADDR + index + offsetof(env_t, data),
Wolfgang Denkea882ba2010-06-20 23:33:59 +020026 1, (char *)&c);
Igor Grinberg0901d9f2011-11-07 01:14:02 +000027 return c;
wdenk5779d8d2003-12-06 23:55:10 +000028}
29
Wolfgang Denkea882ba2010-06-20 23:33:59 +020030void env_relocate_spec(void)
wdenk5779d8d2003-12-06 23:55:10 +000031{
Tom Rinicd0f4fa2013-04-05 14:55:21 -040032 char buf[CONFIG_ENV_SIZE];
Wolfgang Denkea882ba2010-06-20 23:33:59 +020033
Tom Rinicd0f4fa2013-04-05 14:55:21 -040034 read_dataflash(CONFIG_ENV_ADDR, CONFIG_ENV_SIZE, buf);
35
36 env_import(buf, 1);
wdenk5779d8d2003-12-06 23:55:10 +000037}
38
Wolfgang Denkea882ba2010-06-20 23:33:59 +020039#ifdef CONFIG_ENV_OFFSET_REDUND
40#error No support for redundant environment on dataflash yet!
41#endif
42
wdenk5779d8d2003-12-06 23:55:10 +000043int saveenv(void)
44{
Tom Rinicd0f4fa2013-04-05 14:55:21 -040045 env_t env_new;
Wolfgang Denkea882ba2010-06-20 23:33:59 +020046 ssize_t len;
47 char *res;
48
Tom Rinicd0f4fa2013-04-05 14:55:21 -040049 res = (char *)&env_new.data;
Joe Hershbergerbe112352012-12-11 22:16:23 -060050 len = hexport_r(&env_htab, '\0', 0, &res, ENV_SIZE, 0, NULL);
Wolfgang Denkea882ba2010-06-20 23:33:59 +020051 if (len < 0) {
52 error("Cannot export environment: errno = %d\n", errno);
53 return 1;
54 }
Tom Rinicd0f4fa2013-04-05 14:55:21 -040055 env_new.crc = crc32(0, env_new.data, ENV_SIZE);
Wolfgang Denkea882ba2010-06-20 23:33:59 +020056
57 return write_dataflash(CONFIG_ENV_ADDR,
Tom Rinicd0f4fa2013-04-05 14:55:21 -040058 (unsigned long)&env_new,
Wolfgang Denkea882ba2010-06-20 23:33:59 +020059 CONFIG_ENV_SIZE);
wdenk5779d8d2003-12-06 23:55:10 +000060}
61
Wolfgang Denkea882ba2010-06-20 23:33:59 +020062/*
63 * Initialize environment use
wdenk5779d8d2003-12-06 23:55:10 +000064 *
Wolfgang Denkea882ba2010-06-20 23:33:59 +020065 * We are still running from ROM, so data use is limited.
wdenk5779d8d2003-12-06 23:55:10 +000066 * Use a (moderately small) buffer on the stack
67 */
68int env_init(void)
69{
Igor Grinberg0901d9f2011-11-07 01:14:02 +000070 ulong crc, len = ENV_SIZE, new = 0;
wdenk5779d8d2003-12-06 23:55:10 +000071 unsigned off;
72 uchar buf[64];
wdenk5779d8d2003-12-06 23:55:10 +000073
Wolfgang Denkea882ba2010-06-20 23:33:59 +020074 if (gd->env_valid)
75 return 0;
76
77 AT91F_DataflashInit(); /* prepare for DATAFLASH read/write */
78
79 /* read old CRC */
80 read_dataflash(CONFIG_ENV_ADDR + offsetof(env_t, crc),
81 sizeof(ulong), (char *)&crc);
82
Igor Grinberg0901d9f2011-11-07 01:14:02 +000083 off = offsetof(env_t, data);
Wolfgang Denkea882ba2010-06-20 23:33:59 +020084 while (len > 0) {
85 int n = (len > sizeof(buf)) ? sizeof(buf) : len;
86
87 read_dataflash(CONFIG_ENV_ADDR + off, n, (char *)buf);
88
Igor Grinberg0901d9f2011-11-07 01:14:02 +000089 new = crc32(new, buf, n);
Wolfgang Denkea882ba2010-06-20 23:33:59 +020090 len -= n;
91 off += n;
wdenk5779d8d2003-12-06 23:55:10 +000092 }
93
Wolfgang Denkea882ba2010-06-20 23:33:59 +020094 if (crc == new) {
Igor Grinberg0901d9f2011-11-07 01:14:02 +000095 gd->env_addr = offsetof(env_t, data);
96 gd->env_valid = 1;
Wolfgang Denkea882ba2010-06-20 23:33:59 +020097 } else {
Igor Grinberg0901d9f2011-11-07 01:14:02 +000098 gd->env_addr = (ulong)&default_environment[0];
99 gd->env_valid = 0;
Wolfgang Denkea882ba2010-06-20 23:33:59 +0200100 }
101
102 return 0;
wdenk5779d8d2003-12-06 23:55:10 +0000103}