autcpu12.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:7k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /*
  2.  *  drivers/mtd/autcpu12.c
  3.  *
  4.  *  Copyright (c) 2001 Thomas Gleixner (gleixner@autronix.de)
  5.  *
  6.  *  Derived from drivers/mtd/spia.c
  7.  *   Copyright (C) 2000 Steven J. Hill (sjhill@cotw.com)
  8.  * 
  9.  * $Id: autcpu12.c,v 1.3 2002/02/26 17:58:24 gleixner Exp $
  10.  *
  11.  * This program is free software; you can redistribute it and/or modify
  12.  * it under the terms of the GNU General Public License version 2 as
  13.  * published by the Free Software Foundation.
  14.  *
  15.  *  Overview:
  16.  *   This is a device driver for the NAND flash device found on the
  17.  *   autronix autcpu12 board, which is a SmartMediaCard. It supports 
  18.  *   16MB, 32MB and 64MB cards.
  19.  *
  20.  * 02-12-2002 TG Cleanup of module params
  21.  *
  22.  * 02-20-2002 TG adjusted for different rd/wr adress support
  23.  * added support for read device ready/busy line
  24.  * added page_cache
  25.  */
  26. #include <linux/slab.h>
  27. #include <linux/module.h>
  28. #include <linux/mtd/mtd.h>
  29. #include <linux/mtd/nand.h>
  30. #include <linux/mtd/partitions.h>
  31. #include <asm/io.h>
  32. #include <asm/arch/hardware.h>
  33. #include <asm/sizes.h>
  34. #include <asm/arch/autcpu12.h>
  35. /*
  36.  * MTD structure for AUTCPU12 board
  37.  */
  38. static struct mtd_info *autcpu12_mtd = NULL;
  39. /*
  40.  * Module stuff
  41.  */
  42. #if LINUX_VERSION_CODE < 0x20212 && defined(MODULE)
  43. #define autcpu12_init init_module
  44. #define autcpu12_cleanup cleanup_module
  45. #endif
  46. static int autcpu12_io_base = CS89712_VIRT_BASE;
  47. static int autcpu12_fio_pbase = AUTCPU12_PHYS_SMC;
  48. static int autcpu12_fio_ctrl = AUTCPU12_SMC_SELECT_OFFSET;
  49. static int autcpu12_pedr = AUTCPU12_SMC_PORT_OFFSET;
  50. static int autcpu12_fio_base;
  51. #ifdef MODULE
  52. MODULE_PARM(autcpu12_fio_pbase, "i");
  53. MODULE_PARM(autcpu12_fio_ctrl, "i");
  54. MODULE_PARM(autcpu12_pedr, "i");
  55. __setup("autcpu12_fio_pbase=",autcpu12_fio_pbase);
  56. __setup("autcpu12_fio_ctrl=",autcpu12_fio_ctrl);
  57. __setup("autcpu12_pedr=",autcpu12_pedr);
  58. #endif
  59. /*
  60.  * Define partitions for flash devices
  61.  */
  62. static struct mtd_partition partition_info16k[] = {
  63. { name: "AUTCPU12 flash partition 1",
  64.   offset:  0,
  65.   size:    8 * SZ_1M },
  66. { name: "AUTCPU12 flash partition 2",
  67.   offset:  8 * SZ_1M,
  68.   size:    8 * SZ_1M },
  69. };
  70. static struct mtd_partition partition_info32k[] = {
  71. { name: "AUTCPU12 flash partition 1",
  72.   offset:  0,
  73.   size:   16 * SZ_1M },
  74. { name: "AUTCPU12 flash partition 2",
  75.   offset: 16 * SZ_1M,
  76.   size:   24 * SZ_1M },
  77. };
  78. static struct mtd_partition partition_info64k[] = {
  79. { name: "AUTCPU12 flash partition 1",
  80.   offset:  0,
  81.   size:   16 * SZ_1M },
  82. { name: "AUTCPU12 flash partition 2",
  83.   offset: 16 * SZ_1M,
  84.   size:   48 * SZ_1M},
  85. };
  86. #define NUM_PARTITIONS16K 2
  87. #define NUM_PARTITIONS32K 3
  88. #define NUM_PARTITIONS64K 4
  89. /* 
  90.  * hardware specific access to control-lines
  91. */
  92. void autcpu12_hwcontrol(int cmd)
  93. {
  94. switch(cmd){
  95. case NAND_CTL_SETCLE: (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) |=  AUTCPU12_SMC_CLE; break;
  96. case NAND_CTL_CLRCLE: (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) &= ~AUTCPU12_SMC_CLE; break;
  97. case NAND_CTL_SETALE: (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) |=  AUTCPU12_SMC_ALE; break;
  98. case NAND_CTL_CLRALE: (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) &= ~AUTCPU12_SMC_ALE; break;
  99. case NAND_CTL_SETNCE: (*(volatile unsigned char *) (autcpu12_fio_base + autcpu12_fio_ctrl)) = 0x01; break;
  100. case NAND_CTL_CLRNCE: (*(volatile unsigned char *) (autcpu12_fio_base + autcpu12_fio_ctrl)) = 0x00; break;
  101. }
  102. }
  103. /*
  104. * read device ready pin
  105. */
  106. int autcpu12_device_ready(void)
  107. {
  108. return ( (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) & AUTCPU12_SMC_RDY) ? 1 : 0;
  109. }
  110. /*
  111.  * Main initialization routine
  112.  */
  113. int __init autcpu12_init (void)
  114. {
  115. struct nand_chip *this;
  116. int err = 0;
  117. /* Allocate memory for MTD device structure and private data */
  118. autcpu12_mtd = kmalloc (sizeof(struct mtd_info) + sizeof (struct nand_chip),
  119. GFP_KERNEL);
  120. if (!autcpu12_mtd) {
  121. printk ("Unable to allocate AUTCPU12 NAND MTD device structure.n");
  122. err = -ENOMEM;
  123. goto out;
  124. }
  125. /* map physical adress */
  126. autcpu12_fio_base=(unsigned long)ioremap(autcpu12_fio_pbase,SZ_1K);
  127. if(!autcpu12_fio_base){
  128. printk("Ioremap autcpu12 SmartMedia Card failedn");
  129. err = -EIO;
  130. goto out_mtd;
  131. }
  132. /* Get pointer to private data */
  133. this = (struct nand_chip *) (&autcpu12_mtd[1]);
  134. /* Initialize structures */
  135. memset((char *) autcpu12_mtd, 0, sizeof(struct mtd_info));
  136. memset((char *) this, 0, sizeof(struct nand_chip));
  137. /* Link the private data with the MTD structure */
  138. autcpu12_mtd->priv = this;
  139. /* Set address of NAND IO lines */
  140. this->IO_ADDR_R = autcpu12_fio_base;
  141. this->IO_ADDR_W = autcpu12_fio_base;
  142. this->hwcontrol = autcpu12_hwcontrol;
  143. this->dev_ready = autcpu12_device_ready;
  144. /* 20 us command delay time */
  145. this->chip_delay = 20;
  146. /* Scan to find existance of the device */
  147. if (nand_scan (autcpu12_mtd)) {
  148. err = -ENXIO;
  149. goto out_ior;
  150. }
  151. /* Allocate memory for internal data buffer */
  152. this->data_buf = kmalloc (sizeof(u_char) * (autcpu12_mtd->oobblock + autcpu12_mtd->oobsize), GFP_KERNEL);
  153. if (!this->data_buf) {
  154. printk ("Unable to allocate NAND data buffer for AUTCPU12.n");
  155. err = -ENOMEM;
  156. goto out_ior;
  157. }
  158. /* Allocate memory for internal data buffer */
  159. this->data_cache = kmalloc (sizeof(u_char) * (autcpu12_mtd->oobblock + autcpu12_mtd->oobsize), GFP_KERNEL);
  160. if (!this->data_cache) {
  161. printk ("Unable to allocate NAND data cache for AUTCPU12.n");
  162. err = -ENOMEM;
  163. goto out_buf;
  164. }
  165. this->cache_page = -1;
  166. /* Register the partitions */
  167. switch(autcpu12_mtd->size){
  168. case SZ_16M: add_mtd_partitions(autcpu12_mtd, partition_info16k, NUM_PARTITIONS16K); break;
  169. case SZ_32M: add_mtd_partitions(autcpu12_mtd, partition_info32k, NUM_PARTITIONS32K); break;
  170. case SZ_64M: add_mtd_partitions(autcpu12_mtd, partition_info64k, NUM_PARTITIONS64K); break; 
  171. default: {
  172. printk ("Unsupported SmartMedia devicen"); 
  173. err = -ENXIO;
  174. goto out_cac;
  175. }
  176. }
  177. goto out;
  178. out_cac:
  179. kfree (this->data_cache);    
  180. out_buf:
  181. kfree (this->data_buf);    
  182. out_ior:
  183. iounmap((void *)autcpu12_fio_base);
  184. out_mtd:
  185. kfree (autcpu12_mtd);
  186. out:
  187. return err;
  188. }
  189. module_init(autcpu12_init);
  190. /*
  191.  * Clean up routine
  192.  */
  193. #ifdef MODULE
  194. static void __exit autcpu12_cleanup (void)
  195. {
  196. struct nand_chip *this = (struct nand_chip *) &autcpu12_mtd[1];
  197. /* Unregister partitions */
  198. del_mtd_partitions(autcpu12_mtd);
  199. /* Unregister the device */
  200. del_mtd_device (autcpu12_mtd);
  201. /* Free internal data buffers */
  202. kfree (this->data_buf);
  203. kfree (this->data_cache);
  204. /* unmap physical adress */
  205. iounmap((void *)autcpu12_fio_base);
  206. /* Free the MTD device structure */
  207. kfree (autcpu12_mtd);
  208. }
  209. module_exit(autcpu12_cleanup);
  210. #endif
  211. MODULE_LICENSE("GPL");
  212. MODULE_AUTHOR("Thomas Gleixner <gleixner@autronix.de>");
  213. MODULE_DESCRIPTION("Glue layer for SmartMediaCard on autronix autcpu12");