~ubuntu-branches/ubuntu/maverick/u-boot-omap3/maverick

« back to all changes in this revision

Viewing changes to board/mx1ads/syncflash.c

  • Committer: Bazaar Package Importer
  • Author(s): Oliver Grawert
  • Date: 2010-03-22 15:06:23 UTC
  • Revision ID: james.westby@ubuntu.com-20100322150623-i21g8rgiyl5dohag
Tags: upstream-2010.3git20100315
ImportĀ upstreamĀ versionĀ 2010.3git20100315

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/*
 
2
 * board/mx1ads/syncflash.c
 
3
 *
 
4
 * (c) Copyright 2004
 
5
 * Techware Information Technology, Inc.
 
6
 * http://www.techware.com.tw/
 
7
 *
 
8
 * Ming-Len Wu <minglen_wu@techware.com.tw>
 
9
 *
 
10
 * This program is free software; you can redistribute it and/or
 
11
 * modify it under the terms of the GNU General Public License as
 
12
 * published by the Free Software Foundation; either version 2 of
 
13
 * the License, or (at your option) any later version.
 
14
 *
 
15
 * This program is distributed in the hope that it will be useful,
 
16
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 
17
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
18
 * GNU General Public License for more details.
 
19
 *
 
20
 * You should have received a copy of the GNU General Public License
 
21
 * along with this program; if not, write to the Free Software
 
22
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
 
23
 * MA 02111-1307 USA
 
24
 */
 
25
 
 
26
#include <common.h>
 
27
/*#include <mc9328.h>*/
 
28
#include <asm/arch/imx-regs.h>
 
29
 
 
30
typedef unsigned long * p_u32;
 
31
 
 
32
/* 4Mx16x2 IAM=0 CSD1 */
 
33
 
 
34
flash_info_t flash_info[CONFIG_SYS_MAX_FLASH_BANKS];    /* info for FLASH chips    */
 
35
 
 
36
/*  Following Setting is for CSD1       */
 
37
#define SFCTL                   0x00221004
 
38
#define reg_SFCTL               __REG(SFCTL)
 
39
 
 
40
#define SYNCFLASH_A10           (0x00100000)
 
41
 
 
42
#define CMD_NORMAL              (0x81020300)                    /* Normal Mode                  */
 
43
#define CMD_PREC                (CMD_NORMAL + 0x10000000)       /* Precharge Command            */
 
44
#define CMD_AUTO                (CMD_NORMAL + 0x20000000)       /* Auto Refresh Command         */
 
45
#define CMD_LMR                 (CMD_NORMAL + 0x30000000)       /* Load Mode Register Command   */
 
46
#define CMD_LCR                 (CMD_NORMAL + 0x60000000)       /* LCR Command                  */
 
47
#define CMD_PROGRAM             (CMD_NORMAL + 0x70000000)
 
48
 
 
49
#define MODE_REG_VAL            (CONFIG_SYS_FLASH_BASE+0x0008CC00)      /* Cas Latency 3                */
 
50
 
 
51
/* LCR Command */
 
52
#define LCR_READSTATUS          (0x0001C000)                    /* 0x70                         */
 
53
#define LCR_ERASE_CONFIRM       (0x00008000)                    /* 0x20                         */
 
54
#define LCR_ERASE_NVMODE        (0x0000C000)                    /* 0x30                         */
 
55
#define LCR_PROG_NVMODE         (0x00028000)                    /* 0xA0                         */
 
56
#define LCR_SR_CLEAR            (0x00014000)                    /* 0x50                         */
 
57
 
 
58
/* Get Status register                  */
 
59
u32 SF_SR(void) {
 
60
        u32 tmp,tmp1;
 
61
 
 
62
        reg_SFCTL       = CMD_PROGRAM;
 
63
        tmp             = __REG(CONFIG_SYS_FLASH_BASE);
 
64
 
 
65
        reg_SFCTL       = CMD_NORMAL;
 
66
 
 
67
        reg_SFCTL       = CMD_LCR;                      /* Activate LCR Mode            */
 
68
        tmp1            = __REG(CONFIG_SYS_FLASH_BASE + LCR_SR_CLEAR);
 
69
 
 
70
        return tmp;
 
71
}
 
72
 
 
73
/* check if SyncFlash is ready          */
 
74
u8 SF_Ready(void) {
 
75
        u32 tmp;
 
76
 
 
77
        tmp     = SF_SR();
 
78
 
 
79
        if ((tmp & 0x00800000) && (tmp & 0x001C0000)) {
 
80
                printf ("SyncFlash Error code %08x\n",tmp);
 
81
        };
 
82
 
 
83
        if ((tmp & 0x00000080) && (tmp & 0x0000001C)) {
 
84
                printf ("SyncFlash Error code %08x\n",tmp);
 
85
        };
 
86
 
 
87
        if (tmp == 0x00800080)          /* Test Bit 7 of SR     */
 
88
                return 1;
 
89
        else
 
90
                return 0;
 
91
}
 
92
 
 
93
/* Issue the precharge all command              */
 
94
void SF_PrechargeAll(void) {
 
95
 
 
96
        u32 tmp;
 
97
 
 
98
        reg_SFCTL       = CMD_PREC;                     /* Set Precharge Command        */
 
99
        tmp             = __REG(CONFIG_SYS_FLASH_BASE + SYNCFLASH_A10); /* Issue Precharge All Command */
 
100
}
 
101
 
 
102
/* set SyncFlash to normal mode                 */
 
103
void SF_Normal(void) {
 
104
 
 
105
        SF_PrechargeAll();
 
106
 
 
107
        reg_SFCTL       = CMD_NORMAL;
 
108
}
 
109
 
 
110
/* Erase SyncFlash                              */
 
111
void SF_Erase(u32 RowAddress) {
 
112
        u32 tmp;
 
113
 
 
114
        reg_SFCTL       = CMD_NORMAL;
 
115
        tmp             = __REG(RowAddress);
 
116
 
 
117
        reg_SFCTL       = CMD_PREC;
 
118
        tmp             = __REG(RowAddress);
 
119
 
 
120
        reg_SFCTL       = CMD_LCR;                      /* Set LCR mode         */
 
121
        __REG(RowAddress + LCR_ERASE_CONFIRM)   = 0;    /* Issue Erase Setup Command    */
 
122
 
 
123
        reg_SFCTL       = CMD_NORMAL;                   /* return to Normal mode        */
 
124
        __REG(RowAddress)       = 0xD0D0D0D0;           /* Confirm                      */
 
125
 
 
126
        while(!SF_Ready());
 
127
}
 
128
 
 
129
void SF_NvmodeErase(void) {
 
130
        SF_PrechargeAll();
 
131
 
 
132
        reg_SFCTL       = CMD_LCR;                      /* Set to LCR mode              */
 
133
        __REG(CONFIG_SYS_FLASH_BASE + LCR_ERASE_NVMODE)  = 0;   /* Issue Erase Nvmode Reg Command */
 
134
 
 
135
        reg_SFCTL       = CMD_NORMAL;                   /* Return to Normal mode        */
 
136
        __REG(CONFIG_SYS_FLASH_BASE + LCR_ERASE_NVMODE) = 0xC0C0C0C0;   /* Confirm              */
 
137
 
 
138
        while(!SF_Ready());
 
139
}
 
140
 
 
141
void SF_NvmodeWrite(void) {
 
142
        SF_PrechargeAll();
 
143
 
 
144
        reg_SFCTL       = CMD_LCR;                      /* Set to LCR mode              */
 
145
        __REG(CONFIG_SYS_FLASH_BASE+LCR_PROG_NVMODE) = 0;       /* Issue Program Nvmode reg command */
 
146
 
 
147
        reg_SFCTL       = CMD_NORMAL;                   /* Return to Normal mode        */
 
148
        __REG(CONFIG_SYS_FLASH_BASE+LCR_PROG_NVMODE) = 0xC0C0C0C0;      /* Confirm not needed   */
 
149
}
 
150
 
 
151
/****************************************************************************************/
 
152
 
 
153
ulong flash_init(void) {
 
154
        int i, j;
 
155
        u32 tmp;
 
156
 
 
157
/* Turn on CSD1 for negating RESETSF of SyncFLash */
 
158
 
 
159
        reg_SFCTL       |= 0x80000000;          /* enable CSD1 for SyncFlash            */
 
160
        udelay(200);
 
161
 
 
162
        reg_SFCTL       = CMD_LMR;              /* Set Load Mode Register Command       */
 
163
        tmp             = __REG(MODE_REG_VAL);  /* Issue Load Mode Register Command     */
 
164
 
 
165
        SF_Normal();
 
166
 
 
167
        i = 0;
 
168
 
 
169
        flash_info[i].flash_id  =  FLASH_MAN_MT | FLASH_MT28S4M16LC;
 
170
 
 
171
        flash_info[i].size      = FLASH_BANK_SIZE;
 
172
        flash_info[i].sector_count = CONFIG_SYS_MAX_FLASH_SECT;
 
173
 
 
174
        memset(flash_info[i].protect, 0, CONFIG_SYS_MAX_FLASH_SECT);
 
175
 
 
176
        for (j = 0; j < flash_info[i].sector_count; j++) {
 
177
                flash_info[i].start[j] = CONFIG_SYS_FLASH_BASE + j * 0x00100000;
 
178
        }
 
179
 
 
180
        flash_protect(FLAG_PROTECT_SET,
 
181
                CONFIG_SYS_FLASH_BASE,
 
182
                CONFIG_SYS_FLASH_BASE + monitor_flash_len - 1,
 
183
                &flash_info[0]);
 
184
 
 
185
        flash_protect(FLAG_PROTECT_SET,
 
186
                CONFIG_ENV_ADDR,
 
187
                CONFIG_ENV_ADDR + CONFIG_ENV_SIZE - 1,
 
188
                &flash_info[0]);
 
189
 
 
190
        return FLASH_BANK_SIZE;
 
191
}
 
192
 
 
193
void flash_print_info (flash_info_t *info) {
 
194
 
 
195
        int i;
 
196
 
 
197
        switch (info->flash_id & FLASH_VENDMASK) {
 
198
                case (FLASH_MAN_MT & FLASH_VENDMASK):
 
199
                        printf("Micron: ");
 
200
                        break;
 
201
                default:
 
202
                        printf("Unknown Vendor ");
 
203
                        break;
 
204
        }
 
205
 
 
206
        switch (info->flash_id & FLASH_TYPEMASK) {
 
207
                case (FLASH_MT28S4M16LC & FLASH_TYPEMASK):
 
208
                        printf("2x FLASH_MT28S4M16LC (16MB Total)\n");
 
209
                        break;
 
210
                default:
 
211
                        printf("Unknown Chip Type\n");
 
212
                        return;
 
213
                        break;
 
214
        }
 
215
 
 
216
        printf("  Size: %ld MB in %d Sectors\n",
 
217
                info->size >> 20, info->sector_count);
 
218
 
 
219
        printf("  Sector Start Addresses: ");
 
220
 
 
221
        for (i = 0; i < info->sector_count; i++) {
 
222
                if ((i % 5) == 0)
 
223
                        printf ("\n   ");
 
224
 
 
225
                printf (" %08lX%s", info->start[i],
 
226
                        info->protect[i] ? " (RO)" : "     ");
 
227
        }
 
228
 
 
229
        printf ("\n");
 
230
}
 
231
 
 
232
/*-----------------------------------------------------------------------*/
 
233
 
 
234
int flash_erase (flash_info_t *info, int s_first, int s_last) {
 
235
        int iflag, cflag, prot, sect;
 
236
        int rc = ERR_OK;
 
237
 
 
238
/* first look for protection bits */
 
239
 
 
240
        if (info->flash_id == FLASH_UNKNOWN)
 
241
                return ERR_UNKNOWN_FLASH_TYPE;
 
242
 
 
243
        if ((s_first < 0) || (s_first > s_last))
 
244
                return ERR_INVAL;
 
245
 
 
246
        if ((info->flash_id & FLASH_VENDMASK) != (FLASH_MAN_MT & FLASH_VENDMASK))
 
247
                return ERR_UNKNOWN_FLASH_VENDOR;
 
248
 
 
249
        prot = 0;
 
250
 
 
251
        for (sect = s_first; sect <= s_last; ++sect) {
 
252
                if (info->protect[sect])
 
253
                        prot++;
 
254
        }
 
255
 
 
256
        if (prot) {
 
257
                printf("protected!\n");
 
258
                return ERR_PROTECTED;
 
259
        }
 
260
/*
 
261
 * Disable interrupts which might cause a timeout
 
262
 * here. Remember that our exception vectors are
 
263
 * at address 0 in the flash, and we don't want a
 
264
 * (ticker) exception to happen while the flash
 
265
 * chip is in programming mode.
 
266
 */
 
267
 
 
268
        cflag = icache_status();
 
269
        icache_disable();
 
270
        iflag = disable_interrupts();
 
271
 
 
272
/* Start erase on unprotected sectors */
 
273
        for (sect = s_first; sect <= s_last && !ctrlc(); sect++) {
 
274
 
 
275
                printf("Erasing sector %2d ... ", sect);
 
276
 
 
277
/* arm simple, non interrupt dependent timer */
 
278
 
 
279
                reset_timer_masked();
 
280
 
 
281
                SF_NvmodeErase();
 
282
                SF_NvmodeWrite();
 
283
 
 
284
                SF_Erase(CONFIG_SYS_FLASH_BASE + (0x0100000 * sect));
 
285
                SF_Normal();
 
286
 
 
287
                printf("ok.\n");
 
288
        }
 
289
 
 
290
        if (ctrlc())
 
291
                printf("User Interrupt!\n");
 
292
 
 
293
        if (iflag)
 
294
                enable_interrupts();
 
295
 
 
296
        if (cflag)
 
297
                icache_enable();
 
298
 
 
299
        return rc;
 
300
}
 
301
 
 
302
/*-----------------------------------------------------------------------
 
303
 * Copy memory to flash.
 
304
 */
 
305
 
 
306
int write_buff (flash_info_t *info, uchar *src, ulong addr, ulong cnt) {
 
307
        int i;
 
308
 
 
309
        for(i = 0; i < cnt; i += 4) {
 
310
 
 
311
                SF_PrechargeAll();
 
312
 
 
313
                reg_SFCTL       = CMD_PROGRAM;          /* Enter SyncFlash Program mode */
 
314
                __REG(addr + i) = __REG((u32)src  + i);
 
315
 
 
316
                while(!SF_Ready());
 
317
        }
 
318
 
 
319
        SF_Normal();
 
320
 
 
321
        return ERR_OK;
 
322
}