1*10465441SEvalZero /*
2*10465441SEvalZero * JFFS2 -- Journalling Flash File System, Version 2.
3*10465441SEvalZero *
4*10465441SEvalZero * Copyright (C) 2001-2003 Red Hat, Inc.
5*10465441SEvalZero *
6*10465441SEvalZero * Created by Dominic Ostrowski <[email protected]>
7*10465441SEvalZero * Contributors: David Woodhouse, Nick Garnett, Richard Panton.
8*10465441SEvalZero *
9*10465441SEvalZero * For licensing information, see the file 'LICENCE' in this directory.
10*10465441SEvalZero *
11*10465441SEvalZero * $Id: flashio.c,v 1.1 2003/11/26 14:09:29 dwmw2 Exp $
12*10465441SEvalZero *
13*10465441SEvalZero */
14*10465441SEvalZero
15*10465441SEvalZero #include <linux/kernel.h>
16*10465441SEvalZero #include "nodelist.h"
17*10465441SEvalZero #include <rtdevice.h>
18*10465441SEvalZero
jffs2_flash_read(struct jffs2_sb_info * c,cyg_uint32 offset,const size_t size,size_t * return_size,unsigned char * buffer)19*10465441SEvalZero int jffs2_flash_read(struct jffs2_sb_info * c, cyg_uint32 offset,
20*10465441SEvalZero const size_t size,
21*10465441SEvalZero size_t * return_size,
22*10465441SEvalZero unsigned char *buffer)
23*10465441SEvalZero {
24*10465441SEvalZero uint32_t len;
25*10465441SEvalZero struct super_block *sb = OFNI_BS_2SFFJ(c);
26*10465441SEvalZero
27*10465441SEvalZero len = rt_mtd_nor_read(RT_MTD_NOR_DEVICE(sb->s_dev), offset, buffer, size);
28*10465441SEvalZero if (len != size)
29*10465441SEvalZero return -EIO;
30*10465441SEvalZero
31*10465441SEvalZero * return_size = len;
32*10465441SEvalZero return ENOERR;
33*10465441SEvalZero }
34*10465441SEvalZero
jffs2_flash_write(struct jffs2_sb_info * c,cyg_uint32 offset,const size_t size,size_t * return_size,unsigned char * buffer)35*10465441SEvalZero int jffs2_flash_write(struct jffs2_sb_info * c,
36*10465441SEvalZero cyg_uint32 offset, const size_t size,
37*10465441SEvalZero size_t * return_size, unsigned char *buffer)
38*10465441SEvalZero {
39*10465441SEvalZero uint32_t len;
40*10465441SEvalZero struct super_block *sb = OFNI_BS_2SFFJ(c);
41*10465441SEvalZero
42*10465441SEvalZero len = rt_mtd_nor_write(RT_MTD_NOR_DEVICE(sb->s_dev), offset, buffer, size);
43*10465441SEvalZero if (len != size)
44*10465441SEvalZero return -EIO;
45*10465441SEvalZero
46*10465441SEvalZero * return_size = len;
47*10465441SEvalZero return ENOERR;
48*10465441SEvalZero }
49*10465441SEvalZero
jffs2_flash_erase(struct jffs2_sb_info * c,struct jffs2_eraseblock * jeb)50*10465441SEvalZero int jffs2_flash_erase(struct jffs2_sb_info * c,
51*10465441SEvalZero struct jffs2_eraseblock * jeb)
52*10465441SEvalZero {
53*10465441SEvalZero rt_err_t result;
54*10465441SEvalZero struct super_block *sb = OFNI_BS_2SFFJ(c);
55*10465441SEvalZero
56*10465441SEvalZero result = rt_mtd_nor_erase_block(RT_MTD_NOR_DEVICE(sb->s_dev), jeb->offset, c->sector_size);
57*10465441SEvalZero if (result != RT_EOK)
58*10465441SEvalZero return -EIO;
59*10465441SEvalZero
60*10465441SEvalZero return ENOERR;
61*10465441SEvalZero }
62*10465441SEvalZero
jffs2_flash_direct_writev(struct jffs2_sb_info * c,const struct iovec * vecs,unsigned long count,loff_t to,size_t * retlen)63*10465441SEvalZero int jffs2_flash_direct_writev(struct jffs2_sb_info *c,
64*10465441SEvalZero const struct iovec *vecs, unsigned long count, loff_t to,
65*10465441SEvalZero size_t * retlen)
66*10465441SEvalZero {
67*10465441SEvalZero unsigned long i;
68*10465441SEvalZero size_t totlen = 0, thislen;
69*10465441SEvalZero int ret = 0;
70*10465441SEvalZero
71*10465441SEvalZero for (i = 0; i < count; i++)
72*10465441SEvalZero {
73*10465441SEvalZero // writes need to be aligned but the data we're passed may not be
74*10465441SEvalZero // Observation suggests most unaligned writes are small, so we
75*10465441SEvalZero // optimize for that case.
76*10465441SEvalZero
77*10465441SEvalZero if (((vecs[i].iov_len & (sizeof(int) - 1))) ||
78*10465441SEvalZero (((unsigned long) vecs[i].iov_base & (sizeof(unsigned long) - 1))))
79*10465441SEvalZero {
80*10465441SEvalZero // are there iov's after this one? Or is it so much we'd need
81*10465441SEvalZero // to do multiple writes anyway?
82*10465441SEvalZero if ((i + 1) < count || vecs[i].iov_len > 256)
83*10465441SEvalZero {
84*10465441SEvalZero // cop out and malloc
85*10465441SEvalZero unsigned long j;
86*10465441SEvalZero ssize_t sizetomalloc = 0, totvecsize = 0;
87*10465441SEvalZero char *cbuf, *cbufptr;
88*10465441SEvalZero
89*10465441SEvalZero for (j = i; j < count; j++)
90*10465441SEvalZero totvecsize += vecs[j].iov_len;
91*10465441SEvalZero
92*10465441SEvalZero // pad up in case unaligned
93*10465441SEvalZero sizetomalloc = totvecsize + sizeof(int) - 1;
94*10465441SEvalZero sizetomalloc &= ~(sizeof(int) - 1);
95*10465441SEvalZero cbuf = (char *) rt_malloc(sizetomalloc);
96*10465441SEvalZero // malloc returns aligned memory
97*10465441SEvalZero if (!cbuf)
98*10465441SEvalZero {
99*10465441SEvalZero ret = -ENOMEM;
100*10465441SEvalZero goto writev_out;
101*10465441SEvalZero }
102*10465441SEvalZero cbufptr = cbuf;
103*10465441SEvalZero for (j = i; j < count; j++)
104*10465441SEvalZero {
105*10465441SEvalZero memcpy(cbufptr, vecs[j].iov_base, vecs[j].iov_len);
106*10465441SEvalZero cbufptr += vecs[j].iov_len;
107*10465441SEvalZero }
108*10465441SEvalZero //rt_kprintf("direct_write: offset %d, size %d\n", to, sizetomalloc);
109*10465441SEvalZero ret = jffs2_flash_write(c, to, sizetomalloc, &thislen,
110*10465441SEvalZero (unsigned char *) cbuf);
111*10465441SEvalZero if (thislen > totvecsize) // in case it was aligned up
112*10465441SEvalZero thislen = totvecsize;
113*10465441SEvalZero totlen += thislen;
114*10465441SEvalZero rt_free(cbuf);
115*10465441SEvalZero goto writev_out;
116*10465441SEvalZero }
117*10465441SEvalZero else
118*10465441SEvalZero {
119*10465441SEvalZero // otherwise optimize for the common case
120*10465441SEvalZero int buf[256/sizeof(int)]; // int, so int aligned
121*10465441SEvalZero size_t lentowrite;
122*10465441SEvalZero
123*10465441SEvalZero lentowrite = vecs[i].iov_len;
124*10465441SEvalZero // pad up in case its unaligned
125*10465441SEvalZero lentowrite += sizeof(int) - 1;
126*10465441SEvalZero lentowrite &= ~(sizeof(int) - 1);
127*10465441SEvalZero memcpy(buf, vecs[i].iov_base, lentowrite);
128*10465441SEvalZero
129*10465441SEvalZero //rt_kprintf("direct_write: offset %d, size %d\n", to, lentowrite);
130*10465441SEvalZero ret = jffs2_flash_write(c, to, lentowrite, &thislen,
131*10465441SEvalZero (unsigned char *) &buf);
132*10465441SEvalZero if (thislen > vecs[i].iov_len)
133*10465441SEvalZero thislen = vecs[i].iov_len;
134*10465441SEvalZero }
135*10465441SEvalZero }
136*10465441SEvalZero else
137*10465441SEvalZero {
138*10465441SEvalZero //rt_kprintf("direct_writev: offset %d, size %d\n", to, vecs[i].iov_len);
139*10465441SEvalZero ret = jffs2_flash_write(c, to, vecs[i].iov_len, &thislen,
140*10465441SEvalZero vecs[i].iov_base);
141*10465441SEvalZero }
142*10465441SEvalZero totlen += thislen;
143*10465441SEvalZero if (ret || thislen != vecs[i].iov_len) break;
144*10465441SEvalZero to += vecs[i].iov_len;
145*10465441SEvalZero }
146*10465441SEvalZero
147*10465441SEvalZero writev_out:
148*10465441SEvalZero if (retlen) *retlen = totlen;
149*10465441SEvalZero
150*10465441SEvalZero return ret;
151*10465441SEvalZero }
152