summaryrefslogtreecommitdiff
path: root/os/cerf405/nand.c
diff options
context:
space:
mode:
authorCharles.Forsyth <devnull@localhost>2006-12-22 21:39:35 +0000
committerCharles.Forsyth <devnull@localhost>2006-12-22 21:39:35 +0000
commit74a4d8c26dd3c1e9febcb717cfd6cb6512991a7a (patch)
treec6e220ba61db3a6ea4052e6841296d829654e664 /os/cerf405/nand.c
parent46439007cf417cbd9ac8049bb4122c890097a0fa (diff)
20060303
Diffstat (limited to 'os/cerf405/nand.c')
-rw-r--r--os/cerf405/nand.c96
1 files changed, 96 insertions, 0 deletions
diff --git a/os/cerf405/nand.c b/os/cerf405/nand.c
new file mode 100644
index 00000000..5567574c
--- /dev/null
+++ b/os/cerf405/nand.c
@@ -0,0 +1,96 @@
+#include "u.h"
+#include "../port/lib.h"
+#include "mem.h"
+#include "dat.h"
+#include "fns.h"
+#include "io.h"
+#include "../port/error.h"
+
+#include "flashif.h"
+
+/*
+ * Cerf405-specific NAND flash interface
+ */
+
+#define BE(n) (1<<(31-(n))) /* big-endian bit numbering */
+
+enum {
+ /* GPIO lines */
+ Gpio_CLE_o_b= 31,
+ Gpio_ALE_o_b= 30,
+ Gpio_NCE_o_b= 24, /* CE#, active low */
+ Gpio_RDY_i_b= 23,
+
+ /* bit masks */
+ Gpio_CLE_o= BE(Gpio_CLE_o_b),
+ Gpio_ALE_o= BE(Gpio_ALE_o_b),
+ Gpio_NCE_o= BE(Gpio_NCE_o_b),
+ Gpio_RDY_i= BE(Gpio_RDY_i_b),
+
+ Gpio_NAND_o= Gpio_CLE_o | Gpio_ALE_o | Gpio_NCE_o,
+
+ CS_NAND= 1,
+ Gpio_PerCS1_o= BE(10),
+};
+
+void
+archnand_init(Flash*)
+{
+ gpioreserve(Gpio_NAND_o | Gpio_RDY_i);
+ gpioset(Gpio_NAND_o, Gpio_NCE_o);
+ gpioconfig(Gpio_NAND_o, Gpio_out);
+ gpioconfig(Gpio_RDY_i, Gpio_in);
+}
+
+void
+archnand_claim(Flash*, int claim)
+{
+ gpioset(Gpio_NCE_o, claim? 0: Gpio_NCE_o);
+}
+
+void
+archnand_setCLEandALE(Flash*, int cle, int ale)
+{
+ ulong v;
+
+ v = 0;
+ if(cle)
+ v |= Gpio_CLE_o;
+ if(ale)
+ v |= Gpio_ALE_o;
+ gpioset(Gpio_CLE_o | Gpio_ALE_o, v);
+}
+
+/*
+ * could unroll the loops
+ */
+
+void
+archnand_read(Flash *f, void *buf, int len)
+{
+ uchar *p, *bp;
+
+ p = f->addr;
+ if(buf != nil){
+ bp = buf;
+ while(--len >= 0)
+ *bp++ = *p;
+ }else{
+ int junk;
+ while(--len >= 0){
+ junk = *p;
+ USED(junk);
+ }
+ }
+}
+
+void
+archnand_write(Flash *f, void *buf, int len)
+{
+ uchar *p, *bp;
+
+ p = f->addr;
+ bp = buf;
+ while(--len >= 0)
+ *p = *bp++;
+}