Chinaunix首页 | 论坛 | 博客
  • 博客访问: 3192781
  • 博文数量: 685
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 5303
  • 用 户 组: 普通用户
  • 注册时间: 2014-04-19 14:17
个人简介

文章分类

全部博文(685)

文章存档

2015年(116)

2014年(569)

分类: 嵌入式

2014-09-20 13:58:19

/*
 * makeyaffs2image.c
 *
 * Makes a YAFFS2 file system image that can be used to load up a file system.
 * Uses default Linux MTD layout - search for "NAND LAYOUT" to change.
 */
 
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "yaffs_ecc.h"
#include "yaffs_guts.h"

#include "yaffs_packedtags2.h"

unsigned yaffs_trace_mask=0;

#define MAX_OBJECTS 10000

// Adjust these to match your NAND LAYOUT:
#define chunkSize 2048//chunk erea sie per page(pageSize=chunk+spare)
#define spareSize 64//spare erea size per page
#define pagesPerBlock 64//how many pages per block

 

typedef struct
{
 dev_t dev;
 ino_t ino;
 int   obj;
} objItem;


static objItem obj_list[MAX_OBJECTS];
static int obj_id = YAFFS_NOBJECT_BUCKETS + 1;

static int n_obj, nDirectories, nPages;

static int outFile;

static int error;
static int savedErrno;

static int convert_endian = 0;

static void fatal(const char *fn)
{
 perror(fn);
 error |= 1;
 exit(error);
}

static int warn(const char *fn)
{
 savedErrno = errno;
 perror(fn);
 error |= 2;
 return error;
}

static int obj_compare(const void *a, const void * b)
{
  objItem *oa, *ob;
 
  oa = (objItem *)a;
  ob = (objItem *)b;
 
  if(oa->dev < ob->dev) return -1;
  if(oa->dev > ob->dev) return 1;
  if(oa->ino < ob->ino) return -1;
  if(oa->ino > ob->ino) return 1;
 
  return 0;
}


static void add_obj_to_list(dev_t dev, ino_t ino, int obj)
{
 if(n_obj < MAX_OBJECTS)
 {
  obj_list[n_obj].dev = dev;
  obj_list[n_obj].ino = ino;
  obj_list[n_obj].obj = obj;
  n_obj++;
  qsort(obj_list,n_obj,sizeof(objItem),obj_compare);
  
 }
 else
 {
  // oops! not enough space in the object array
  fprintf(stderr,"Not enough space in object array\n");
  exit(1);
 }
}


static int find_obj_in_list(dev_t dev, ino_t ino)
{
 objItem *i = NULL;
 objItem test;

 test.dev = dev;
 test.ino = ino;
 
 if(n_obj > 0)
 {
  i = bsearch(&test,obj_list,n_obj,sizeof(objItem),obj_compare);
 }

 if(i)
 {
  return i->obj;
 }
 return -1;
}

/* This little function converts a little endian tag to a big endian tag.
 * NOTE: The tag is not usable after this other than calculating the CRC
 * with.
 */
static void little_to_big_endian(struct yaffs_ext_tags *tagsPtr)
{
#if 0 // FIXME NCB
    union yaffs_tags_union * tags = (union yaffs_tags_union* )tagsPtr; // Work in bytes.
    union yaffs_tags_union   temp;

    memset(&temp, 0, sizeof(temp));
    // Ick, I hate magic numbers.
    temp.as_bytes[0] = ((tags->as_bytes[2] & 0x0F) << 4) | ((tags->as_bytes[1] & 0xF0) >> 4);
    temp.as_bytes[1] = ((tags->as_bytes[1] & 0x0F) << 4) | ((tags->as_bytes[0] & 0xF0) >> 4);
    temp.as_bytes[2] = ((tags->as_bytes[0] & 0x0F) << 4) | ((tags->as_bytes[2] & 0x30) >> 2) | ((tags->as_bytes[3] & 0xC0) >> 6);
    temp.as_bytes[3] = ((tags->as_bytes[3] & 0x3F) << 2) | ((tags->as_bytes[2] & 0xC0) >> 6);
    temp.as_bytes[4] = ((tags->as_bytes[6] & 0x03) << 6) | ((tags->as_bytes[5] & 0xFC) >> 2);
    temp.as_bytes[5] = ((tags->as_bytes[5] & 0x03) << 6) | ((tags->as_bytes[4] & 0xFC) >> 2);
    temp.as_bytes[6] = ((tags->as_bytes[4] & 0x03) << 6) | (tags->as_bytes[7] & 0x3F);
    temp.as_bytes[7] = (tags->as_bytes[6] & 0xFC) | ((tags->as_bytes[7] & 0xC0) >> 6);

    // Now copy it back.
    tags->as_bytes[0] = temp.as_bytes[0];
    tags->as_bytes[1] = temp.as_bytes[1];
    tags->as_bytes[2] = temp.as_bytes[2];
    tags->as_bytes[3] = temp.as_bytes[3];
    tags->as_bytes[4] = temp.as_bytes[4];
    tags->as_bytes[5] = temp.as_bytes[5];
    tags->as_bytes[6] = temp.as_bytes[6];
    tags->as_bytes[7] = temp.as_bytes[7];
#endif
}

static void shuffle_oob(char *spareData, struct yaffs_packed_tags2 *pt)
{
 assert(sizeof(*pt) <= spareSize);
 // NAND LAYOUT: For non-trivial OOB orderings, here would be a good place to shuffle.
 memcpy(spareData, pt, sizeof(*pt));
}

/*
struct yaffs_packed_tags2 {
    struct yaffs_packed_tags2_tags_only t;     //数据
    struct yaffs_ecc_other ecc;                       //校验信息
};
yaffs_packed_tags2是由数据和校难信息两部分组成的,但是这儿只用了数据不进行校验,所以内核里面也只是读出数据不校验
*/

static int write_chunk(u8 *data, u32 id, u32 chunk_id, u32 n_bytes)
{
 struct yaffs_ext_tags t;
 struct yaffs_packed_tags2 pt;
 char spareData[spareSize];

 if (write(outFile,data,chunkSize) != chunkSize)//一次将Page页中chunk区域全部写满。
  fatal("write");

 memset(&t, 0, sizeof(t));
 
 t.chunk_id = chunk_id;
// t.serial_number = 0;
 t.serial_number = 1; // **CHECK**
 t.n_bytes = n_bytes;
 t.obj_id = id;
 
 t.seq_number = YAFFS_LOWEST_SEQUENCE_NUMBER;

// added NCB **CHECK**
 t.chunk_used = 1;

 if (convert_endian)
 {
         little_to_big_endian(&t);
 }

 nPages++;

 memset(&pt, 0, sizeof(pt));
 yaffs_pack_tags2(&pt,&t,1);
 //函数将 yaffs_ext_tags转为yaffs_packed_tags2,并生成校验信息,但最最后一个参数为0(tag_ecc标志位),所以只保存转换的tag信息并不进行ecc校验更不会保存校验信息。
 memset(spareData, 0xff, sizeof(spareData));
 shuffle_oob(spareData, &pt);

 if (write(outFile,spareData,sizeof(spareData)) != sizeof(spareData))
  fatal("write");//将flash上oob区域的数据写到映像中(oob即spare区域).
 return 0;
}

#define SWAP32(x)   ((((x) & 0x000000FF) << 24) | \
                     (((x) & 0x0000FF00) << 8 ) | \
                     (((x) & 0x00FF0000) >> 8 ) | \
                     (((x) & 0xFF000000) >> 24))

#define SWAP16(x)   ((((x) & 0x00FF) << 8) | \
                     (((x) & 0xFF00) >> 8))
       
// This one is easier, since the types are more standard. No funky shifts here.
static void object_header_little_to_big_endian(struct yaffs_obj_hdr* oh)
{
    oh->type = SWAP32(oh->type); // GCC makes enums 32 bits.
    oh->parent_obj_id = SWAP32(oh->parent_obj_id); // int
    oh->sum_no_longer_used = SWAP16(oh->sum_no_longer_used); // u16 - Not used, but done for completeness.
    // name = skip. Char array. Not swapped.
    oh->yst_mode = SWAP32(oh->yst_mode);

    // Regular POSIX.
    oh->yst_uid = SWAP32(oh->yst_uid);
    oh->yst_gid = SWAP32(oh->yst_gid);
    oh->yst_atime = SWAP32(oh->yst_atime);
    oh->yst_mtime = SWAP32(oh->yst_mtime);
    oh->yst_ctime = SWAP32(oh->yst_ctime);

    oh->file_size_low = SWAP32(oh->file_size_low); // Aiee. An int... signed, at that!
    oh->file_size_high = SWAP32(oh->file_size_high); // Aiee. An int... signed, at that!
    oh->equiv_id = SWAP32(oh->equiv_id);
    // alias  - char array.
    oh->yst_rdev = SWAP32(oh->yst_rdev);

    oh->win_ctime[0] = SWAP32(oh->win_ctime[0]);
    oh->win_ctime[1] = SWAP32(oh->win_ctime[1]);
    oh->win_atime[0] = SWAP32(oh->win_atime[0]);
    oh->win_atime[1] = SWAP32(oh->win_atime[1]);
    oh->win_mtime[0] = SWAP32(oh->win_mtime[0]);
    oh->win_mtime[1] = SWAP32(oh->win_mtime[1]);

    oh->reserved[0] = SWAP32(oh->reserved[0]);
    oh->reserved[1] = SWAP32(oh->reserved[1]);

    oh->inband_shadowed_obj_id = SWAP32(oh->inband_shadowed_obj_id);
    oh->inband_is_shrink = SWAP32(oh->inband_is_shrink);
    oh->shadows_obj = SWAP32(oh->shadows_obj);
    oh->is_shrink = SWAP32(oh->is_shrink);
}


static int write_object_header(int id, enum yaffs_obj_type t, struct stat *s, int parent, const char *name, int equivalentObj, const char * alias)
{
 u8 bytes[chunkSize];
 
 struct yaffs_obj_hdr *oh = (struct yaffs_obj_hdr *)bytes;
 
 memset(bytes,0xff,sizeof(bytes));
 
 oh->type = t;

 oh->parent_obj_id = parent;
 
 if (strlen(name)+1 > sizeof(oh->name))
 {
  errno = ENAMETOOLONG;
  return warn("object name");
 }
 memset(oh->name,0,sizeof(oh->name));
 strcpy(oh->name,name);
 
 
 if(t != YAFFS_OBJECT_TYPE_HARDLINK)
 {
  oh->yst_mode = s->st_mode;
  oh->yst_uid = s->st_uid;
// NCB 12/9/02  oh->yst_gid = s->yst_uid;
  oh->yst_gid = s->st_gid;
  oh->yst_atime = s->st_atime;
  oh->yst_mtime = s->st_mtime;
  oh->yst_ctime = s->st_ctime;
  oh->yst_rdev  = s->st_rdev;
 }
 
 if(t == YAFFS_OBJECT_TYPE_FILE)
 {
  oh->file_size_low = s->st_size;
  oh->file_size_high = (s->st_size >> 32);
 }
 
 if(t == YAFFS_OBJECT_TYPE_HARDLINK)
 {
  oh->equiv_id = equivalentObj;
 }
 
 if(t == YAFFS_OBJECT_TYPE_SYMLINK)
 {
  if (strlen(alias)+1 > sizeof(oh->alias))
  {
   errno = ENAMETOOLONG;
   return warn("object alias");
  }
  memset(oh->alias,0,sizeof(oh->alias));
  strcpy(oh->alias,alias);
 }

 if (convert_endian)
 {
      object_header_little_to_big_endian(oh);
 }
 
 return write_chunk(bytes,id,0,0xffff);
 
}

static void pad_image(void)
{
 u8 data[chunkSize + spareSize];
 int padPages = (nPages % pagesPerBlock);

 if (padPages)
 {
  memset(data, 0xff, sizeof(data));
  for (padPages = pagesPerBlock-padPages; padPages; padPages--)
  {
   if (write(outFile, data, sizeof(data)) != sizeof(data))
    fatal("write");
  }
 }
}

static int process_directory(int parent, const char *path)
{

 DIR *dir;
 struct dirent *entry;

 nDirectories++;
 
 dir = opendir(path);//根据传入的路径,打开目录
 if(!dir)
 {
  warn("opendir");
 }
 else
 {
  while((entry = readdir(dir)) != NULL)//依次遍历打开目录中的所有子目录
  {
  
   /* Ignore . and .. */
   if(strcmp(entry->d_name,".") &&
      strcmp(entry->d_name,".."))
    {
     char full_name[500];
    struct stat stats;
    int equivalentObj;
    int newObj;
    
    if (snprintf(full_name,sizeof(full_name),"%s/%s",path,entry->d_name) >= (int)sizeof(full_name))
    {//构造文件的路径(目录文件、FIFO管道文件、字符文件、套接字等)
     error = -1;
     continue;
    }
    
    if (lstat(full_name,&stats) < 0)
    {//获取该文件的stat信息。
     warn("lstat");
     continue;
    }
    
    if(S_ISLNK(stats.st_mode) ||
        S_ISREG(stats.st_mode) ||
        S_ISDIR(stats.st_mode) ||
        S_ISFIFO(stats.st_mode) ||
        S_ISBLK(stats.st_mode) ||
        S_ISCHR(stats.st_mode) ||
        S_ISSOCK(stats.st_mode))
    {//若该文件是链接文件、目录文件、字符文件...会处理,其他类型则不处理
    
     newObj = obj_id++;
     n_obj++;
     
     printf("Object %d, %s is a ",newObj,full_name);
     
     /* We're going to create an object for it */
     if((equivalentObj = find_obj_in_list(stats.st_dev, stats.st_ino)) > 0)
     {
       /* we need to make a hard link */
       printf("hard link to object %d\n",equivalentObj);
      write_object_header(newObj, YAFFS_OBJECT_TYPE_HARDLINK, &stats, parent, entry->d_name, equivalentObj, NULL);
     }
     else
     {
      
      add_obj_to_list(stats.st_dev,stats.st_ino,newObj);
      
      if(S_ISLNK(stats.st_mode))
      {
     
       char symname[500];
      
       memset(symname,0, sizeof(symname));
     
       if (readlink(full_name,symname,sizeof(symname) -1) < 0)
       {
        warn("readlink");
       }
       else
       {
        printf("symlink to \"%s\"\n",symname);
        write_object_header(newObj, YAFFS_OBJECT_TYPE_SYMLINK, &stats, parent, entry->d_name, -1, symname);
       }
      }
      else if(S_ISREG(stats.st_mode))//如果该文件是普通文件
      {
       printf("file, ");
       if(write_object_header(newObj, YAFFS_OBJECT_TYPE_FILE, &stats, parent, entry->d_name, -1, NULL) == 0)
       {//先写入文件头。
        int h;
        u8 bytes[chunkSize];
        int n_bytes;
        int chunk = 0;
        
        h = open(full_name,O_RDONLY);
        if(h >= 0)
        {
         memset(bytes,0xff,sizeof(bytes));
         while((n_bytes = read(h,bytes,sizeof(bytes))) > 0)
         {
          chunk++;
          write_chunk(bytes,newObj,chunk,n_bytes);//再写入文件内容
          memset(bytes,0xff,sizeof(bytes));
         }
         if(n_bytes < 0)
            warn("read");
           
         printf("%d data chunks written\n",chunk);
         close(h);
        }
        else
        {
         warn("open");
        }
        
       }       
              
      }
      else if(S_ISSOCK(stats.st_mode))
      {//如果是套接字文件、fifo管道文件、字符文件、块设备文件或目录文件,则只写入文件头即可。
       printf("socket\n");
       write_object_header(newObj, YAFFS_OBJECT_TYPE_SPECIAL, &stats, parent, entry->d_name, -1, NULL);
      }
      else if(S_ISFIFO(stats.st_mode))
      {
       printf("fifo\n");
       write_object_header(newObj, YAFFS_OBJECT_TYPE_SPECIAL, &stats, parent, entry->d_name, -1, NULL);
      }
      else if(S_ISCHR(stats.st_mode))
      {
       printf("character device\n");
       write_object_header(newObj, YAFFS_OBJECT_TYPE_SPECIAL, &stats, parent, entry->d_name, -1, NULL);
      }
      else if(S_ISBLK(stats.st_mode))
      {
       printf("block device\n");
       write_object_header(newObj, YAFFS_OBJECT_TYPE_SPECIAL, &stats, parent, entry->d_name, -1, NULL);
      }
      else if(S_ISDIR(stats.st_mode))
      {
       printf("directory\n");
       if (write_object_header(newObj, YAFFS_OBJECT_TYPE_DIRECTORY, &stats, parent, entry->d_name, -1, NULL) == 0)
        process_directory(newObj,full_name);
      }//若是目录文件则只写入文件头,并把这次的newObj作为parent继续处理其子目录下的内容
     }
    }
    else
    {
     fprintf(stderr, "%s: unhandled type\n", full_name);
     error |= 2;
     savedErrno = EINVAL;
    }//目前只处理链接文件、FIFO管道文件、目录文件、字符文件等7中文件类型。
   }//如果目录下内容为.或..  则不做任何处理直接进行while的下次循环(即打开下一个子目录)。
  }//传入路径中的所有子目录都已经遍历完毕。
  closedir(dir);
 }
 
 return 0;

}


int main(int argc, char *argv[])
{
 struct stat stats;
 
 printf("mkyaffs2image: image building tool for YAFFS2 built "__DATE__"\n");
 
 if(argc < 3)
 {
  printf("usage: mkyaffs2image dir image_file [convert]\n");
  printf("           dir        the directory tree to be converted\n");
  printf("           image_file the output file to hold the image\n");
        printf("           'convert'  produce a big-endian image from a little-endian machine\n");
  exit(1);
 }

    if ((argc == 4) && (!strncmp(argv[3], "convert", strlen("convert"))))
    {
        convert_endian = 1;
    }
   
 if(stat(argv[1],&stats) < 0)
 {
  printf("Could not stat %s\n",argv[1]);
  exit(1);
 }
 
 if(!S_ISDIR(stats.st_mode))//保证argv[1]必须是一个目录
 {
  printf(" %s is not a directory\n",argv[1]);
  exit(1);
 }
 
 outFile = open(argv[2],O_CREAT | O_TRUNC | O_WRONLY, S_IREAD | S_IWRITE);
 //创建并打开一个文件,该文件的fd句柄赋值给全局变量outFile
 
 if(outFile < 0)
 {
  printf("Could not open output file %s\n",argv[2]);
  exit(1);
 }
 
 printf("Processing directory %s into image file %s\n",argv[1],argv[2]);
 process_directory(YAFFS_OBJECTID_ROOT,argv[1]);//将rootfs目录中的所有内容写到outFile中
 
 pad_image();//将映像文件扩充到block对齐。

 close(outFile);
 
 if(error)
 {
  errno = savedErrno;
  perror("operation incomplete");
 }
 else
 {
  printf("Operation complete.\n"
         "%d objects in %d directories\n"
         "%d NAND pages\n",n_obj, nDirectories, nPages);
 }
 
 exit(error);

阅读(1785) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~