A20sata热拔插的实现


From caaadb4287017cda8eed8ea01c7c5bb3ba5f37b1 Mon Sep 17 00:00:00 2001
From: wesage <[email protected]>
Date: Tue, 13 Aug 2013 21:48:28 +0800
Subject: [PATCH 12/40] Fix SATA_OK,201308132148,wesage

---
.../device/softwinner/wing-k70/init.sun7i.rc       |    1 +
.../base/core/res/res/xml/storage_list.xml         |    8 +
android4.2/device/softwinner/wing-k70/vold.fstab   |    1 +
lichee/linux-3.3/.config                           |   34 +-
.../arch/arm/mach-sun7i/include/mach/sys_config.h  |  103 ++
lichee/linux-3.3/arch/arm/mach-sun7i/sys_config.c  | 1708 +++++++++++++++++++-
lichee/linux-3.3/drivers/ata/ahci.h                |    6 +
lichee/linux-3.3/drivers/ata/libahci.c             |   13 +-
lichee/linux-3.3/drivers/ata/sw_ahci_platform.c    |  215 +--
lichee/linux-3.3/drivers/ata/sw_ahci_platform.h    |   32 +-
.../sun7i/configs/android/wing-k70/sys_config.fex  |    2 +-
11 files changed, 1996 insertions(+), 127 deletions(-)

diff --git a/android4.2/device/softwinner/wing-k70/init.sun7i.rc b/android4.2/device/softwinner/wing-k70/init.sun7i.rc
index 8ebfdd6..012b656 100755
--- a/android4.2/device/softwinner/wing-k70/init.sun7i.rc
+++ b/android4.2/device/softwinner/wing-k70/init.sun7i.rc
@@ -9,6 +9,7 @@ on early-init
        mkdir /mnt/extsd 0000 system system
        mkdir /mnt/usbhost1 0000 system system
        mkdir /databk 0700 root  system
+        mkdir /mnt/sata 0000 system system
        
on early-fs
        mount ext4 /dev/block/system /system   wait noatime nodev barrier=0
diff --git a/android4.2/device/softwinner/wing-k70/overlay/frameworks/base/core/res/res/xml/storage_list.xml b/android4.2/device/softwinner/wing-k70/overlay/frameworks/base/core/res/res/xml/storage_list.xml
index 64e994b..5c19610 100755
--- a/android4.2/device/softwinner/wing-k70/overlay/frameworks/base/core/res/res/xml/storage_list.xml
+++ b/android4.2/device/softwinner/wing-k70/overlay/frameworks/base/core/res/res/xml/storage_list.xml
@@ -56,5 +56,13 @@
              android:emulated="false"  
              android:mtpReserve="0" 
              android:maxFileSize="0"/>
+        <storage android:mountPoint="/mnt/sata"
+             android:storageDescription="sata"
+             android:primary="false"
+             android:removable="true"
+             android:emulated="false"  
+             android:mtpReserve="0" 
+             android:allowMassStorage="true"
+             android:maxFileSize="0"/>
</StorageList>

diff --git a/android4.2/device/softwinner/wing-k70/vold.fstab b/android4.2/device/softwinner/wing-k70/vold.fstab
index 3034cea..9482dba 100755
--- a/android4.2/device/softwinner/wing-k70/vold.fstab
+++ b/android4.2/device/softwinner/wing-k70/vold.fstab
@@ -17,3 +17,4 @@
dev_mount        sdcard        /mnt/sdcard        auto        /devices/virtual/block/nandj        /devices/platform/sunxi-mmc.2/mmc_host
dev_mount        extsd        /mnt/extsd        auto        /devices/platform/sunxi-mmc.1/mmc_host        /devices/platform/sunxi-mmc.0/mmc_host
dev_mount        usbhost1        /mnt/usbhost1        auto        /devices/platform/sw-ehci.1        /devices/platform/sw_hcd_host0        /devices/platform/sw-ohci.1

+dev_mount        sata        /mnt/sata        auto        /devices/platform/sw_ahci.0/ata1/host0

diff --git a/lichee/linux-3.3/.config b/lichee/linux-3.3/.config

index dbca6ed..663e61c 100644
--- a/lichee/linux-3.3/.config
+++ b/lichee/linux-3.3/.config
@@ -970,7 +970,39 @@ CONFIG_SCSI_LOWLEVEL=y
# CONFIG_SCSI_DEBUG is not set
# CONFIG_SCSI_DH is not set
# CONFIG_SCSI_OSD_INITIATOR is not set
-# CONFIG_ATA is not set
+CONFIG_ATA=y
+# CONFIG_ATA_NONSTANDARD is not set
+CONFIG_ATA_VERBOSE_ERROR=y
+CONFIG_SATA_PMP=y
+
+#
+# Controllers with non-SFF native interface
+#
+CONFIG_SATA_AHCI_PLATFORM=y
+CONFIG_SW_SATA_AHCI_PLATFORM=y
+CONFIG_ATA_SFF=y
+
+#
+# SFF controllers with custom DMA interface
+#
+CONFIG_ATA_BMDMA=y
+
+#
+# SATA SFF controllers with BMDMA
+#
+# CONFIG_SATA_MV is not set
+
+#
+# PATA SFF controllers with BMDMA
+#
+
+#
+# PIO-only SFF controllers
+#
+
+#
+# Generic fallback / legacy drivers
+#
CONFIG_MD=y
# CONFIG_BLK_DEV_MD is not set
CONFIG_BLK_DEV_DM=y
diff --git a/lichee/linux-3.3/arch/arm/mach-sun7i/include/mach/sys_config.h b/lichee/linux-3.3/arch/arm/mach-sun7i/include/mach/sys_config.h
index 1f50e9d..1dac71b 100644
--- a/lichee/linux-3.3/arch/arm/mach-sun7i/include/mach/sys_config.h
+++ b/lichee/linux-3.3/arch/arm/mach-sun7i/include/mach/sys_config.h
@@ -18,6 +18,36 @@

#include "gpio.h"

+#define   SYS_CONFIG_MEMBASE                 (PLAT_PHYS_OFFSET + SZ_32M + SZ_16M)
+#define   SYS_CONFIG_MEMSIZE                 (SZ_64K)
+#define   SCRIPT_PARSER_OK                   (0)
+#define   SCRIPT_PARSER_EMPTY_BUFFER         (-1)
+#define   SCRIPT_PARSER_KEYNAME_NULL         (-2)
+#define   SCRIPT_PARSER_DATA_VALUE_NULL      (-3)
+#define   SCRIPT_PARSER_KEY_NOT_FIND         (-4)
+#define   SCRIPT_PARSER_BUFFER_NOT_ENOUGH    (-5)
+
+#define   EGPIO_FAIL             (-1)
+#define   EGPIO_SUCCESS          (0)
+
+
+/*
+ * define script sub key type, raw from sys_config.bin
+ * @SCIRPT_PARSER_VALUE_TYPE_INVALID: invalid type
+ * @SCIRPT_PARSER_VALUE_TYPE_SINGLE_WORD: int item type
+ * @SCIRPT_PARSER_VALUE_TYPE_STRING: string item type
+ * @SCIRPT_PARSER_VALUE_TYPE_MULTI_WORD: multi int type, not used currently
+ * @SCIRPT_PARSER_VALUE_TYPE_GPIO_WORD: gpio item type
+ */
+typedef enum {
+    SCIRPT_PARSER_VALUE_TYPE_INVALID = 0,
+    SCIRPT_PARSER_VALUE_TYPE_SINGLE_WORD,
+    SCIRPT_PARSER_VALUE_TYPE_STRING,
+    SCIRPT_PARSER_VALUE_TYPE_MULTI_WORD,
+    SCIRPT_PARSER_VALUE_TYPE_GPIO_WORD
+} script_parser_value_type_t;
+
+
/*
  * define types of script item
  * @SCIRPT_ITEM_VALUE_TYPE_INVALID:  invalid item type
@@ -45,6 +75,50 @@ typedef union {
     struct gpio_config  gpio;
} script_item_u;

+typedef struct
+{
+    char  gpio_name[32];
+    int port;
+    int port_num;
+    int mul_sel;
+    int pull;
+    int drv_level;
+    int data;
+} user_gpio_set_t;
+
+
+typedef struct
+{
+        char  gpio_name[32];
+        int port;
+        int port_num;
+        int mul_sel;
+        int pull;
+        int drv_level;
+        int data;
+} script_gpio_set_t;
+
+typedef struct
+{
+        int  main_key_count;
+        int  version[3];
+} script_head_t_ex;
+
+typedef struct
+{
+        char main_name[32];
+        int  lenth;
+        int  offset;
+} script_main_key_t_ex;
+
+typedef struct
+{
+        char sub_name[32];
+        int  offset;
+        int  pattern;
+} script_sub_key_t_ex;
+
+
/*
  * script_get_item
  *      get an item from script based on main_key & sub_key
@@ -74,4 +148,33 @@ int script_get_pio_list(char *main_key, script_item_u **list);
  */
int script_dump_mainkey(char *main_key);

+
+/* script operations */
+extern int script_parser_init(char *script_buf);
+extern int script_parser_exit(void);
+extern int script_parser_fetch(char *main_name, char *sub_name, int value[], int count);
+extern int script_parser_fetch_ex(char *main_name, char *sub_name, int value[],
+               script_parser_value_type_t *type, int count);
+extern int script_parser_subkey_count(char *main_name);
+extern int script_parser_mainkey_count(void);
+extern int script_parser_mainkey_get_gpio_count(char *main_name);
+extern int script_parser_mainkey_get_gpio_cfg(char *main_name, void *gpio_cfg, int gpio_count);
+
+
+
+/* gpio operations */
+extern int gpio_init(void);
+extern int gpio_exit(void);
+extern unsigned gpio_request_se(user_gpio_set_t *gpio_list, unsigned group_count_max);
+extern unsigned gpio_request_ex(char *main_name, const char *sub_name);
+extern int gpio_release(unsigned p_handler, int if_release_to_default_status);
+extern int gpio_get_all_pin_status(unsigned p_handler, user_gpio_set_t *gpio_status, unsigned gpio_count_max, unsigned if_get_from_hardware);
+extern int gpio_get_one_pin_status(unsigned p_handler, user_gpio_set_t *gpio_status, const char *gpio_name, unsigned if_get_from_hardware);
+extern int gpio_set_one_pin_status(unsigned p_handler, user_gpio_set_t *gpio_status, const char *gpio_name, unsigned if_set_to_current_input_status);
+extern int gpio_set_one_pin_io_status(unsigned p_handler, unsigned if_set_to_output_status, const char *gpio_name);
+extern int gpio_set_one_pin_pull(unsigned p_handler, unsigned set_pull_status, const char *gpio_name);
+extern int gpio_set_one_pin_driver_level(unsigned p_handler, unsigned set_driver_level, const char *gpio_name);
+extern int gpio_read_one_pin_value(unsigned p_handler, const char *gpio_name);
+extern int gpio_write_one_pin_value(unsigned p_handler, unsigned value_to_gpio, const char *gpio_name);
+
#endif
diff --git a/lichee/linux-3.3/arch/arm/mach-sun7i/sys_config.c b/lichee/linux-3.3/arch/arm/mach-sun7i/sys_config.c
index 88691a7..e716eb1 100644
--- a/lichee/linux-3.3/arch/arm/mach-sun7i/sys_config.c
+++ b/lichee/linux-3.3/arch/arm/mach-sun7i/sys_config.c
@@ -16,6 +16,7 @@
#include <linux/string.h>
#include <linux/module.h>
#include <linux/kernel.h>
+#include <mach/platform.h>
#include <mach/sys_config.h>
#include <mach/gpio.h>
#include <linux/slab.h>
@@ -94,6 +95,10 @@ typedef struct {
} script_origin_gpio_t;
#pragma pack()

+
+u32     gpio_g_pioMemBase;
+
+
/*
  *===========================================================================================================
  * origin script defined as follows:
@@ -207,21 +212,6 @@ typedef struct {
     void                *next;
} script_main_key_t;

-/*
- * define script sub key type, raw from sys_config.bin
- * @SCIRPT_PARSER_VALUE_TYPE_INVALID: invalid type
- * @SCIRPT_PARSER_VALUE_TYPE_SINGLE_WORD: int item type
- * @SCIRPT_PARSER_VALUE_TYPE_STRING: string item type
- * @SCIRPT_PARSER_VALUE_TYPE_MULTI_WORD: multi int type, not used currently
- * @SCIRPT_PARSER_VALUE_TYPE_GPIO_WORD: gpio item type
- */
-typedef enum {
-    SCIRPT_PARSER_VALUE_TYPE_INVALID = 0,
-    SCIRPT_PARSER_VALUE_TYPE_SINGLE_WORD,
-    SCIRPT_PARSER_VALUE_TYPE_STRING,
-    SCIRPT_PARSER_VALUE_TYPE_MULTI_WORD,
-    SCIRPT_PARSER_VALUE_TYPE_GPIO_WORD
-} script_parser_value_type_t;

static script_main_key_t   *g_script;

@@ -613,3 +603,1691 @@ static int script_debugfs_init(void)
     return 0;
}
core_initcall(script_debugfs_init);
+
+
+/**########################################################################################
+ *
+ *                        Script Operations
+ *
+-#########################################################################################*/
+static  char  *script_mod_buf = NULL; //pointer to first key
+static  int    script_main_key_count = 0;
+
+
+static  int   _test_str_length(char *str)
+{
+    int length = 0;
+
+    while(str[length++])
+    {
+        if(length > 32)
+        {
+            length = 32;
+            break;
+        }
+    }
+
+    return length;
+}
+
+int script_parser_init(char *script_buf)
+{
+    script_head_t_ex   *script_head;
+
+    pr_debug("%s(%d)-%s, script_buf addr is %p:\n",__FILE__,__LINE__,__FUNCTION__, script_buf);
+    if(script_buf)
+    {
+        script_mod_buf = script_buf;
+        script_head = (script_head_t_ex *)script_mod_buf;
+
+        script_main_key_count = script_head->main_key_count;
+
+        pr_debug("succeed: %s(%d)-%s\n",__FILE__,__LINE__,__FUNCTION__);
+        return SCRIPT_PARSER_OK;
+    }
+    else
+    {
+        pr_warning("failed: %s(%d)-%s\n",__FILE__,__LINE__,__FUNCTION__);
+        return SCRIPT_PARSER_EMPTY_BUFFER;
+    }
+}
+
+int script_parser_exit(void)
+{
+    script_mod_buf = NULL;
+    script_main_key_count = 0;
+
+    return SCRIPT_PARSER_OK;
+}
+
+int script_parser_fetch(char *main_name, char *sub_name, int value[], int count)
+{
+    char   main_bkname[32], sub_bkname[32];
+    char   *main_char, *sub_char;
+    script_main_key_t_ex  *main_key = NULL;
+    script_sub_key_t_ex   *sub_key = NULL;
+    int    i, j;
+    int    pattern, word_count;
+
+    pr_debug("enter script parse fetch. \n");
+
+    /* check params */
+    if(!script_mod_buf)
+    {
+        return SCRIPT_PARSER_EMPTY_BUFFER;
+    }
+
+    if((main_name == NULL) || (sub_name == NULL))
+    {
+        return SCRIPT_PARSER_KEYNAME_NULL;
+    }
+
+    if(value == NULL)
+    {
+        return SCRIPT_PARSER_DATA_VALUE_NULL;
+    }
+
+    /* truncate string if size >31 bytes */
+    main_char = main_name;
+    if(_test_str_length(main_name) > 31)
+    {
+        memset(main_bkname, 0, 32);
+        strncpy(main_bkname, main_name, 31);
+        main_char = main_bkname;
+    }
+    sub_char = sub_name;
+    if(_test_str_length(sub_name) > 31)
+    {
+        memset(sub_bkname, 0, 32);
+        strncpy(sub_bkname, sub_name, 31);
+        sub_char = sub_bkname;
+    }
+    pr_debug("gpio: main name is : %s, sub_name is: %s", main_char, sub_char);
+
+    for(i=0;i<script_main_key_count;i++)
+    {
+        main_key = (script_main_key_t_ex *)(script_mod_buf + (sizeof(script_head_t_ex)) + i * sizeof(script_main_key_t_ex));
+        if(strcmp(main_key->main_name, main_char))
+        {
+            continue;
+        }
+        
+        /* now find sub key */
+        for(j=0;j<main_key->lenth;j++)
+        {
+            sub_key = (script_sub_key_t_ex *)(script_mod_buf + (main_key->offset<<2) + (j * sizeof(script_sub_key_t_ex)));
+            if(strcmp(sub_key->sub_name, sub_char))
+            {
+                continue;
+            }
+            pattern    = (sub_key->pattern>>16) & 0xffff; /* get datatype */
+            word_count = (sub_key->pattern>> 0) & 0xffff; /*get count of word */
+            pr_debug("pattern is: 0x%x, word_count is: 0x%x, ", pattern, word_count);
+
+            switch(pattern)
+            {
+                case SCIRPT_PARSER_VALUE_TYPE_SINGLE_WORD:
+                    value[0] = *(int *)(script_mod_buf + (sub_key->offset<<2));
+                    break;
+
+                case SCIRPT_PARSER_VALUE_TYPE_STRING:
+                    if(count < word_count)
+                    {
+                        word_count = count;
+                    }
+                    memcpy((char *)value, script_mod_buf + (sub_key->offset<<2), word_count << 2);
+                    break;
+
+                case SCIRPT_PARSER_VALUE_TYPE_MULTI_WORD:
+                    break;
+                case SCIRPT_PARSER_VALUE_TYPE_GPIO_WORD:
+                {
+                    script_gpio_set_t  *user_gpio_cfg = (script_gpio_set_t *)value;
+                    /* buffer space enough? */
+                    if(sizeof(script_gpio_set_t) > (count<<2))
+                    {
+                        return SCRIPT_PARSER_BUFFER_NOT_ENOUGH;
+                    }
+                    strcpy( user_gpio_cfg->gpio_name, sub_char);
+                    memcpy(&user_gpio_cfg->port, script_mod_buf + (sub_key->offset<<2),  sizeof(script_gpio_set_t) - 32);
+                    break;
+                }
+            }
+
+            return SCRIPT_PARSER_OK;
+        }
+    }
+
+    return SCRIPT_PARSER_KEY_NOT_FIND;
+}
+EXPORT_SYMBOL(script_parser_fetch);
+
+int script_parser_fetch_ex(char *main_name, char *sub_name, int value[], script_parser_value_type_t *type, int count)
+{
+    char   main_bkname[32], sub_bkname[32];
+    char   *main_char, *sub_char;
+    script_main_key_t_ex  *main_key = NULL;
+    script_sub_key_t_ex   *sub_key = NULL;
+    int    i, j;
+    int    pattern, word_count;
+    script_parser_value_type_t *value_type = type;
+
+    pr_debug("enter script parse fetch. \n");
+
+    if(!script_mod_buf)
+    {
+        return SCRIPT_PARSER_EMPTY_BUFFER;
+    }
+
+    if((main_name == NULL) || (sub_name == NULL))
+    {
+        return SCRIPT_PARSER_KEYNAME_NULL;
+    }
+
+    if(value == NULL)
+    {
+        return SCRIPT_PARSER_DATA_VALUE_NULL;
+    }
+
+    main_char = main_name;
+    if(_test_str_length(main_name) > 31)
+    {
+        memset(main_bkname, 0, 32);
+        strncpy(main_bkname, main_name, 31);
+        main_char = main_bkname;
+    }
+    sub_char = sub_name;
+    if(_test_str_length(sub_name) > 31)
+    {
+        memset(sub_bkname, 0, 32);
+        strncpy(sub_bkname, sub_name, 31);
+        sub_char = sub_bkname;
+    }
+    pr_debug("gpio: main name is : %s, sub_name is: %s", main_char, sub_char);
+    
+    for(i=0;i<script_main_key_count;i++)
+    {
+        main_key = (script_main_key_t_ex *)(script_mod_buf + (sizeof(script_head_t_ex)) + i * sizeof(script_main_key_t_ex));
+        if(strcmp(main_key->main_name, main_char))
+        {
+            continue;
+        }
+
+        for(j=0;j<main_key->lenth;j++)
+        {
+            sub_key = (script_sub_key_t_ex *)(script_mod_buf + (main_key->offset<<2) + (j * sizeof(script_sub_key_t_ex)));
+            if(strcmp(sub_key->sub_name, sub_char))
+            {
+                continue;
+            }
+            pattern    = (sub_key->pattern>>16) & 0xffff;
+            word_count = (sub_key->pattern>> 0) & 0xffff;
+            pr_debug("pattern is: 0x%x, word_count is: 0x%x, ", pattern, word_count);
+
+            switch(pattern)
+            {
+                case SCIRPT_PARSER_VALUE_TYPE_SINGLE_WORD:
+                    value[0] = *(int *)(script_mod_buf + (sub_key->offset<<2));
+                    *value_type = SCIRPT_PARSER_VALUE_TYPE_SINGLE_WORD;
+                    break;
+
+                case SCIRPT_PARSER_VALUE_TYPE_STRING:
+                    if(count < word_count)
+                    {
+                        word_count = count;
+                    }
+                    memcpy((char *)value, script_mod_buf + (sub_key->offset<<2), word_count << 2);
+                    *value_type = SCIRPT_PARSER_VALUE_TYPE_STRING;
+                    break;
+
+                case SCIRPT_PARSER_VALUE_TYPE_MULTI_WORD:
+                    *value_type = SCIRPT_PARSER_VALUE_TYPE_MULTI_WORD;
+                    break;
+                case SCIRPT_PARSER_VALUE_TYPE_GPIO_WORD:
+                {
+                    script_gpio_set_t  *user_gpio_cfg = (script_gpio_set_t *)value;
+
+                    if(sizeof(script_gpio_set_t) > (count<<2))
+                    {
+                        return SCRIPT_PARSER_BUFFER_NOT_ENOUGH;
+                    }
+                    strcpy( user_gpio_cfg->gpio_name, sub_char);
+                    memcpy(&user_gpio_cfg->port, script_mod_buf + (sub_key->offset<<2),  sizeof(script_gpio_set_t) - 32);
+                    *value_type = SCIRPT_PARSER_VALUE_TYPE_GPIO_WORD;
+                    break;
+                }
+            }
+
+            return SCRIPT_PARSER_OK;
+        }
+    }
+
+    return SCRIPT_PARSER_KEY_NOT_FIND;
+}
+EXPORT_SYMBOL(script_parser_fetch_ex);
+
+int script_parser_subkey_count(char *main_name)
+{
+    char   main_bkname[32];
+    char   *main_char;
+    script_main_key_t_ex  *main_key = NULL;
+    int    i;
+
+    if(!script_mod_buf)
+    {
+        return SCRIPT_PARSER_EMPTY_BUFFER;
+    }
+
+    if(main_name == NULL)
+    {
+        return SCRIPT_PARSER_KEYNAME_NULL;
+    }
+
+    main_char = main_name;
+    if(_test_str_length(main_name) > 31)
+    {
+        memset(main_bkname, 0, 32);
+        strncpy(main_bkname, main_name, 31);
+        main_char = main_bkname;
+    }
+
+    for(i=0;i<script_main_key_count;i++)
+    {
+        main_key = (script_main_key_t_ex *)(script_mod_buf + (sizeof(script_head_t_ex)) + i * sizeof(script_main_key_t_ex));
+        if(strcmp(main_key->main_name, main_char))
+        {
+            continue;
+        }
+
+        return main_key->lenth;
+    }
+
+    return -1;
+}
+
+int script_parser_mainkey_count(void)
+{
+    if(!script_mod_buf)
+    {
+        return SCRIPT_PARSER_EMPTY_BUFFER;
+    }
+
+    return     script_main_key_count;
+}
+
+int script_parser_mainkey_get_gpio_count(char *main_name)
+{
+    char   main_bkname[32];
+    char   *main_char;
+    script_main_key_t_ex  *main_key = NULL;
+    script_sub_key_t_ex   *sub_key = NULL;
+    int    i, j;
+    int    pattern, gpio_count = 0;
+
+    if(!script_mod_buf)
+    {
+        return SCRIPT_PARSER_EMPTY_BUFFER;
+    }
+
+    if(main_name == NULL)
+    {
+        return SCRIPT_PARSER_KEYNAME_NULL;
+    }
+
+    main_char = main_name;
+    if(_test_str_length(main_name) > 31)
+    {
+        memset(main_bkname, 0, 32);
+        strncpy(main_bkname, main_name, 31);
+        main_char = main_bkname;
+    }
+
+    for(i=0;i<script_main_key_count;i++)
+    {
+        main_key = (script_main_key_t_ex *)(script_mod_buf + (sizeof(script_head_t_ex)) + i * sizeof(script_main_key_t_ex));
+        if(strcmp(main_key->main_name, main_char))
+        {
+            continue;
+        }
+
+        for(j=0;j<main_key->lenth;j++)
+        {
+            sub_key = (script_sub_key_t_ex *)(script_mod_buf + (main_key->offset<<2) + (j * sizeof(script_sub_key_t_ex)));
+
+            pattern    = (sub_key->pattern>>16) & 0xffff;
+
+            if(SCIRPT_PARSER_VALUE_TYPE_GPIO_WORD == pattern)
+            {
+                gpio_count ++;
+            }
+        }
+    }
+
+    return gpio_count;
+}
+EXPORT_SYMBOL(script_parser_mainkey_get_gpio_count);
+
+int script_parser_mainkey_get_gpio_cfg(char *main_name, void *gpio_cfg, int gpio_count)
+{
+    char   main_bkname[32];
+    char   *main_char;
+    script_main_key_t_ex  *main_key = NULL;
+    script_sub_key_t_ex   *sub_key = NULL;
+    script_gpio_set_t  *user_gpio_cfg = (script_gpio_set_t *)gpio_cfg;
+    int    i, j;
+    int    pattern, user_index;
+
+    if(!script_mod_buf)
+    {
+        return SCRIPT_PARSER_EMPTY_BUFFER;
+    }
+
+    if(main_name == NULL)
+    {
+        return SCRIPT_PARSER_KEYNAME_NULL;
+    }
+
+    memset(user_gpio_cfg, 0, sizeof(script_gpio_set_t) * gpio_count);
+
+    main_char = main_name;
+    if(_test_str_length(main_name) > 31)
+    {
+        memset(main_bkname, 0, 32);
+        strncpy(main_bkname, main_name, 31);
+        main_char = main_bkname;
+    }
+
+    for(i=0;i<script_main_key_count;i++)
+    {
+        main_key = (script_main_key_t_ex *)(script_mod_buf + (sizeof(script_head_t_ex)) + i * sizeof(script_main_key_t_ex));
+        if(strcmp(main_key->main_name, main_char))
+        {
+            continue;
+        }
+
+        pr_debug("mainkey name = %s\n", main_key->main_name);
+        user_index = 0;
+        for(j=0;j<main_key->lenth;j++)
+        {
+            sub_key = (script_sub_key_t_ex *)(script_mod_buf + (main_key->offset<<2) + (j * sizeof(script_sub_key_t_ex)));
+            pr_debug("subkey name = %s\n", sub_key->sub_name);
+            pattern    = (sub_key->pattern>>16) & 0xffff;
+            pr_debug("subkey pattern = %d\n", pattern);
+
+            if(SCIRPT_PARSER_VALUE_TYPE_GPIO_WORD == pattern)
+            {
+                strcpy( user_gpio_cfg[user_index].gpio_name, sub_key->sub_name);
+                memcpy(&user_gpio_cfg[user_index].port, script_mod_buf + (sub_key->offset<<2), sizeof(script_gpio_set_t) - 32);
+                user_index++;
+                if(user_index >= gpio_count)
+                {
+                    break;
+                }
+            }
+        }
+        return SCRIPT_PARSER_OK;
+    }
+
+    return SCRIPT_PARSER_KEY_NOT_FIND;
+}
+
+
+
+/**#############################################################################################################
+ *  
+ *                           GPIO(PIN) Operations
+ *
+-##############################################################################################################*/
+#define CSP_OSAL_PHY_2_VIRT(phys, size) SW_VA_PORTC_IO_BASE
+#define CSP_OSAL_MALLOC(size) kmalloc((size), GFP_ATOMIC)
+#define CSP_OSAL_FREE(ptr) kfree((ptr))
+        
+#define    CSP_PIN_PHY_ADDR_BASE    SW_PA_PORTC_IO_BASE
+#define    CSP_PIN_PHY_ADDR_SIZE    0x1000
+        
+        u32                 gpio_g_pioMemBase = 0;
+#define PIOC_REGS_BASE gpio_g_pioMemBase
+        
+        extern char sys_cofig_data[];
+        extern char sys_cofig_data_end[];
+#define __REG(x)                        (*(volatile unsigned int *)(x))
+        
+#define PIO_REG_CFG(n, i)               ((volatile unsigned int *)( PIOC_REGS_BASE + ((n)-1)*0x24 + ((i)<<2) + 0x00))
+#define PIO_REG_DLEVEL(n, i)            ((volatile unsigned int *)( PIOC_REGS_BASE + ((n)-1)*0x24 + ((i)<<2) + 0x14))
+#define PIO_REG_PULL(n, i)              ((volatile unsigned int *)( PIOC_REGS_BASE + ((n)-1)*0x24 + ((i)<<2) + 0x1C))
+#define PIO_REG_DATA(n)                   ((volatile unsigned int *)( PIOC_REGS_BASE + ((n)-1)*0x24 + 0x10))
+        
+#define PIO_REG_CFG_VALUE(n, i)          __REG( PIOC_REGS_BASE + ((n)-1)*0x24 + ((i)<<2) + 0x00)
+#define PIO_REG_DLEVEL_VALUE(n, i)       __REG( PIOC_REGS_BASE + ((n)-1)*0x24 + ((i)<<2) + 0x14)
+#define PIO_REG_PULL_VALUE(n, i)         __REG( PIOC_REGS_BASE + ((n)-1)*0x24 + ((i)<<2) + 0x1C)
+#define PIO_REG_DATA_VALUE(n)            __REG( PIOC_REGS_BASE + ((n)-1)*0x24 + 0x10)
+
+
+typedef struct
+{
+    int mul_sel;
+    int pull;
+    int drv_level;
+    int data;
+} gpio_status_set_t;
+
+typedef struct
+{
+    char    gpio_name[32];
+    int port;
+    int port_num;
+    gpio_status_set_t user_gpio_status;
+    gpio_status_set_t hardware_gpio_status;
+} system_gpio_set_t;
+
+
+/*
+****************************************************************************************************
+*
+*             CSP_PIN_init
+*
+*  Description:
+*       init
+*
+*  Parameters:
+*  Return value:
+*        EGPIO_SUCCESS/EGPIO_FAIL
+****************************************************************************************************
+*/
+int gpio_init(void)
+{
+    printk("Init eGon pin module V2.0\n");
+    gpio_g_pioMemBase = (u32)CSP_OSAL_PHY_2_VIRT(CSP_PIN_PHY_ADDR_BASE , CSP_PIN_PHY_ADDR_SIZE);
+    #ifdef FPGA_RUNTIME_ENV
+    return script_parser_init((char *)(sys_cofig_data));
+    #else
+    return script_parser_init((char *)__va(SYS_CONFIG_MEMBASE));
+    #endif
+}
+fs_initcall(gpio_init);
+
+/*
+****************************************************************************************************
+*
+*             CSP_PIN_exit
+*
+*  Description:
+*       exit
+*
+*  Parameters:
+*
+*  Return value:
+*        EGPIO_SUCCESS/EGPIO_FAIL
+****************************************************************************************************
+*/
+__s32 gpio_exit(void)
+{
+    return 0;
+}
+
+/*
+************************************************************************************************************
+*
+*                                             CSP_GPIO_Request
+*
+*    函数名称:
+*
+*    参数列表:gpio_list      存放所有用到的GPIO数据的数组,GPIO将直接使用这个数组
+*
+*               group_count_max  数组的成员个数,GPIO设定的时候,将操作的GPIO最大不超过这个值
+*
+*    返回值  :
+*
+*    说明    :暂时没有做冲突检查
+*
+*
+************************************************************************************************************
+*/
+u32 gpio_request_se(user_gpio_set_t *gpio_list, __u32 group_count_max)
+{
+    char               *user_gpio_buf;                                        //按照char类型申请
+    system_gpio_set_t  *user_gpio_set, *tmp_sys_gpio_data;                      //user_gpio_set将是申请内存的句柄
+    user_gpio_set_t  *tmp_user_gpio_data;
+    __u32                real_gpio_count = 0, first_port;                      //保存真正有效的GPIO的个数
+    __u32               tmp_group_func_data = 0;
+    __u32               tmp_group_pull_data = 0;
+    __u32               tmp_group_dlevel_data = 0;
+    __u32               tmp_group_data_data = 0;
+    __u32               func_change = 0, pull_change = 0;
+    __u32               dlevel_change = 0, data_change = 0;
+    volatile __u32  *tmp_group_func_addr = NULL, *tmp_group_pull_addr = NULL;
+    volatile __u32  *tmp_group_dlevel_addr = NULL, *tmp_group_data_addr = NULL;
+    __u32  port, port_num, port_num_func, port_num_pull;
+    __u32  pre_port = 0x7fffffff, pre_port_num_func = 0x7fffffff;
+    __u32  pre_port_num_pull = 0x7fffffff;
+    __s32  i, tmp_val;
+
+    if((!gpio_list) || (!group_count_max))
+    {
+        return (u32)0;
+    }
+    for(i = 0; i < group_count_max; i++)
+    {
+        tmp_user_gpio_data = gpio_list + i;                 //gpio_set依次指向每个GPIO数组成员
+        if(!tmp_user_gpio_data->port)
+        {
+            continue;
+        }
+        real_gpio_count ++;
+    }
+
+    //printk("to malloc space for pin \n");
+    user_gpio_buf = (char *)CSP_OSAL_MALLOC(16 + sizeof(system_gpio_set_t) * real_gpio_count);   //申请内存,多申请16个字节,用于存放GPIO个数等信息
+    if(!user_gpio_buf)
+    {
+        return (u32)0;
+    }
+    memset(user_gpio_buf, 0, 16 + sizeof(system_gpio_set_t) * real_gpio_count);         //首先全部清零
+    *(int *)user_gpio_buf = real_gpio_count;                                           //保存有效的GPIO个数
+    user_gpio_set = (system_gpio_set_t *)(user_gpio_buf + 16);                         //指向第一个结构体
+    //准备第一个GPIO数据
+    for(first_port = 0; first_port < group_count_max; first_port++)
+    {
+        tmp_user_gpio_data = gpio_list + first_port;
+        port     = tmp_user_gpio_data->port;                         //读出端口数值
+        port_num = tmp_user_gpio_data->port_num;                     //读出端口中的某一个GPIO
+        if(!port)
+        {
+            continue;
+        }
+        port_num_func = (port_num >> 3);
+        port_num_pull = (port_num >> 4);
+
+        tmp_group_func_addr    = PIO_REG_CFG(port, port_num_func);   //更新功能寄存器地址
+        tmp_group_pull_addr    = PIO_REG_PULL(port, port_num_pull);  //更新pull寄存器
+        tmp_group_dlevel_addr  = PIO_REG_DLEVEL(port, port_num_pull);//更新level寄存器
+        tmp_group_data_addr    = PIO_REG_DATA(port);                 //更新data寄存器
+
+        tmp_group_func_data    = *tmp_group_func_addr;
+        tmp_group_pull_data    = *tmp_group_pull_addr;
+        tmp_group_dlevel_data  = *tmp_group_dlevel_addr;
+        tmp_group_data_data    = *tmp_group_data_addr;
+        break;
+    }
+    if(first_port >= group_count_max)
+    {
+        return 0;
+    }
+    //保存用户数据
+    for(i = first_port; i < group_count_max; i++)
+    {
+        tmp_sys_gpio_data  = user_gpio_set + i;             //tmp_sys_gpio_data指向申请的GPIO空间
+        tmp_user_gpio_data = gpio_list + i;                 //gpio_set依次指向用户的每个GPIO数组成员
+        port     = tmp_user_gpio_data->port;                //读出端口数值
+        port_num = tmp_user_gpio_data->port_num;            //读出端口中的某一个GPIO
+        if(!port)
+        {
+            continue;
+        }
+        //开始保存用户数据
+        strcpy(tmp_sys_gpio_data->gpio_name, tmp_user_gpio_data->gpio_name);
+        tmp_sys_gpio_data->port                       = port;
+        tmp_sys_gpio_data->port_num                   = port_num;
+        tmp_sys_gpio_data->user_gpio_status.mul_sel   = tmp_user_gpio_data->mul_sel;
+        tmp_sys_gpio_data->user_gpio_status.pull      = tmp_user_gpio_data->pull;
+        tmp_sys_gpio_data->user_gpio_status.drv_level = tmp_user_gpio_data->drv_level;
+        tmp_sys_gpio_data->user_gpio_status.data      = tmp_user_gpio_data->data;
+
+        port_num_func = (port_num >> 3);
+        port_num_pull = (port_num >> 4);
+
+        if((port_num_pull != pre_port_num_pull) || (port != pre_port))    //如果发现当前引脚的端口不一致,或者所在的pull寄存器不一致
+        {
+            if(func_change)
+            {
+                *tmp_group_func_addr   = tmp_group_func_data;    //回写功能寄存器
+                func_change = 0;
+            }
+            if(pull_change)
+            {
+                pull_change = 0;
+                *tmp_group_pull_addr   = tmp_group_pull_data;    //回写pull寄存器
+            }
+            if(dlevel_change)
+            {
+                dlevel_change = 0;
+                *tmp_group_dlevel_addr = tmp_group_dlevel_data;  //回写driver level寄存器
+            }
+            if(data_change)
+            {
+                data_change = 0;
+                *tmp_group_data_addr   = tmp_group_data_data;    //回写
+            }
+
+            tmp_group_func_addr    = PIO_REG_CFG(port, port_num_func);   //更新功能寄存器地址
+            tmp_group_pull_addr    = PIO_REG_PULL(port, port_num_pull);   //更新pull寄存器
+            tmp_group_dlevel_addr  = PIO_REG_DLEVEL(port, port_num_pull); //更新level寄存器
+            tmp_group_data_addr    = PIO_REG_DATA(port);                  //更新data寄存器
+                    
+            tmp_group_func_data    = *tmp_group_func_addr;
+            tmp_group_pull_data    = *tmp_group_pull_addr;
+            tmp_group_dlevel_data  = *tmp_group_dlevel_addr;
+            tmp_group_data_data    = *tmp_group_data_addr;
+            
+        }
+        else if(pre_port_num_func != port_num_func)                       //如果发现当前引脚的功能寄存器不一致
+        {
+            *tmp_group_func_addr   = tmp_group_func_data;    //则只回写功能寄存器
+            tmp_group_func_addr    = PIO_REG_CFG(port, port_num_func);   //更新功能寄存器地址
+            
+            tmp_group_func_data    = *tmp_group_func_addr;
+        }
+        //保存当前硬件寄存器数据
+        pre_port_num_pull = port_num_pull;                      //设置当前GPIO成为前一个GPIO
+        pre_port_num_func = port_num_func;
+        pre_port          = port;
+
+        //更新功能寄存器
+        if(tmp_user_gpio_data->mul_sel >= 0)
+        {
+            tmp_val = (port_num - (port_num_func<<3)) << 2;
+            tmp_sys_gpio_data->hardware_gpio_status.mul_sel = (tmp_group_func_data >> tmp_val) & 0x07;
+            tmp_group_func_data &= ~(                              0x07  << tmp_val);
+            tmp_group_func_data |=  (tmp_user_gpio_data->mul_sel & 0x07) << tmp_val;
+            func_change = 1;
+        }
+        //根据pull的值决定是否更新pull寄存器
+
+        tmp_val = (port_num - (port_num_pull<<4)) << 1;
+
+        if(tmp_user_gpio_data->pull >= 0)
+        {
+            tmp_sys_gpio_data->hardware_gpio_status.pull = (tmp_group_pull_data >> tmp_val) & 0x03;
+            if(tmp_user_gpio_data->pull >= 0)
+            {
+                tmp_group_pull_data &= ~(                           0x03  << tmp_val);
+                tmp_group_pull_data |=  (tmp_user_gpio_data->pull & 0x03) << tmp_val;
+                pull_change = 1;
+            }
+        }
+        //根据driver level的值决定是否更新driver level寄存器
+        if(tmp_user_gpio_data->drv_level >= 0)
+        {
+            tmp_sys_gpio_data->hardware_gpio_status.drv_level = (tmp_group_dlevel_data >> tmp_val) & 0x03;
+            if(tmp_user_gpio_data->drv_level >= 0)
+            {
+                tmp_group_dlevel_data &= ~(                                0x03  << tmp_val);
+                tmp_group_dlevel_data |=  (tmp_user_gpio_data->drv_level & 0x03) << tmp_val;
+                dlevel_change = 1;
+            }
+        }
+        //根据用户输入,以及功能分配决定是否更新data寄存器
+        if(tmp_user_gpio_data->mul_sel == 1)
+        {
+            if(tmp_user_gpio_data->data >= 0)
+            {
+                tmp_val = tmp_user_gpio_data->data;
+                tmp_val &= 1;
+                tmp_group_data_data &= ~(1 << port_num);
+                tmp_group_data_data |= tmp_val << port_num;
+                data_change = 1;
+            }
+        }
+    }
+    //for循环结束,如果存在还没有回写的寄存器,这里写回到硬件当中
+    if(tmp_group_func_addr)                         //只要更新过寄存器地址,就可以对硬件赋值
+    {                                               //那么把所有的值全部回写到硬件寄存器
+        *tmp_group_func_addr   = tmp_group_func_data;       //回写功能寄存器
+        if(pull_change)
+        {
+            *tmp_group_pull_addr   = tmp_group_pull_data;    //回写pull寄存器
+        }
+        if(dlevel_change)
+        {
+            *tmp_group_dlevel_addr = tmp_group_dlevel_data;  //回写driver level寄存器
+        }
+        if(data_change)
+        {
+            *tmp_group_data_addr   = tmp_group_data_data;    //回写data寄存器
+        }
+    }
+    
+    return (u32)user_gpio_buf;
+}
+EXPORT_SYMBOL_GPL(gpio_request_se);
+
+/*
+************************************************************************************************************
+*
+*                                             CSP_GPIO_Request_EX
+*
+*    函数名称:
+*
+*    参数说明: main_name   传进的主键名称,匹配模块(驱动名称)
+*
+*               sub_name    传进的子键名称,如果是空,表示全部,否则寻找到匹配的单独GPIO
+*
+*    返回值  :0 :    err
+*              other: success
+*
+*    说明    :暂时没有做冲突检查
+*
+*
+************************************************************************************************************
+*/
+u32 gpio_request_ex(char *main_name, const char *sub_name)  //设备申请GPIO函数扩展接口
+{
+    user_gpio_set_t    *gpio_list=NULL;
+    user_gpio_set_t     one_gpio;
+       __u32               gpio_handle;
+    __s32               gpio_count;
+
+    if(!sub_name){
+            gpio_count = script_parser_mainkey_get_gpio_count(main_name);
+            if(gpio_count <= 0)
+            {
+                printk("err: gpio count < =0 ,gpio_count is: %d \n", gpio_count);
+                return 0;
+            }           
+            gpio_list = (user_gpio_set_t *)CSP_OSAL_MALLOC(sizeof(system_gpio_set_t) * gpio_count); //申请一片临时内存,用于保存用户数据
+            if(!gpio_list){
+            printk("malloc gpio_list error \n");
+                return 0;
+            }
+        if(!script_parser_mainkey_get_gpio_cfg(main_name,gpio_list,gpio_count)){
+            gpio_handle = gpio_request_se(gpio_list, gpio_count);
+            CSP_OSAL_FREE(gpio_list);
+                
+        }else{
+            return 0;
+        }
+        }else{
+            if(script_parser_fetch((char *)main_name, (char *)sub_name, (int *)&one_gpio, (sizeof(user_gpio_set_t) >> 2)) < 0){
+            printk("script parser fetch err. \n");
+            return 0;
+            }
+            
+            gpio_handle = gpio_request_se(&one_gpio, 1);            
+        }
+
+        return gpio_handle;
+}
+EXPORT_SYMBOL(gpio_request_ex);
+
+/*
+****************************************************************************************************
+*
+*             CSP_PIN_DEV_release
+*
+*  Description:
+*       释放某逻辑设备的pin
+*
+*  Parameters:
+*         p_handler    :    handler
+*       if_release_to_default_status : 是否释放到原始状态(寄存器原有状态)
+*
+*  Return value:
+*        EGPIO_SUCCESS/EGPIO_FAIL
+****************************************************************************************************
+*/
+__s32 gpio_release(u32 p_handler, __s32 if_release_to_default_status)
+{
+    char               *tmp_buf;                                        //转换成char类型
+    __u32               group_count_max, first_port;                    //最大GPIO个数
+    system_gpio_set_t  *user_gpio_set, *tmp_sys_gpio_data;
+    __u32               tmp_group_func_data = 0;
+    __u32               tmp_group_pull_data = 0;
+    __u32               tmp_group_dlevel_data = 0;
+    volatile __u32     *tmp_group_func_addr = NULL,   *tmp_group_pull_addr = NULL;
+    volatile __u32     *tmp_group_dlevel_addr = NULL;
+    __u32               port, port_num, port_num_pull, port_num_func;
+    __u32               pre_port = 0x7fffffff, pre_port_num_func = 0x7fffffff, pre_port_num_pull = 0x7fffffff;
+    __u32               i, tmp_val;
+
+    //检查传进的句柄的有效性
+    if(!p_handler)
+    {
+        return EGPIO_FAIL;
+    }
+    tmp_buf = (char *)p_handler;
+    group_count_max = *(int *)tmp_buf;
+    if(!group_count_max)
+    {
+        return EGPIO_FAIL;
+    }
+    if(if_release_to_default_status == 2)
+    {
+        //printk("gpio module :  release p_handler = %x\n",p_handler);
+        CSP_OSAL_FREE((char *)p_handler);
+        
+        return EGPIO_SUCCESS;
+    }
+    user_gpio_set = (system_gpio_set_t *)(tmp_buf + 16);
+    //读取用户数据
+    for(first_port = 0; first_port < group_count_max; first_port++)
+    {
+        tmp_sys_gpio_data  = user_gpio_set + first_port;
+        port     = tmp_sys_gpio_data->port;                 //读出端口数值
+        port_num = tmp_sys_gpio_data->port_num;             //读出端口中的某一个GPIO
+        if(!port)
+        {
+            continue;
+        }
+        port_num_func = (port_num >> 3);
+        port_num_pull = (port_num >> 4);
+
+        tmp_group_func_addr    = PIO_REG_CFG(port, port_num_func);   //更新功能寄存器地址
+        tmp_group_pull_addr    = PIO_REG_PULL(port, port_num_pull);  //更新pull寄存器
+        tmp_group_dlevel_addr  = PIO_REG_DLEVEL(port, port_num_pull);//更新level寄存器
+
+        tmp_group_func_data    = *tmp_group_func_addr;
+        tmp_group_pull_data    = *tmp_group_pull_addr;
+        tmp_group_dlevel_data  = *tmp_group_dlevel_addr;
+        break;
+    }
+    if(first_port >= group_count_max)
+    {
+        return 0;
+    }
+    for(i = first_port; i < group_count_max; i++)
+    {
+        tmp_sys_gpio_data  = user_gpio_set + i;             //tmp_sys_gpio_data指向申请的GPIO空间
+        port     = tmp_sys_gpio_data->port;                 //读出端口数值
+        port_num = tmp_sys_gpio_data->port_num;             //读出端口中的某一个GPIO
+
+        port_num_func = (port_num >> 3);
+        port_num_pull = (port_num >> 4);
+
+        if((port_num_pull != pre_port_num_pull) || (port != pre_port))    //如果发现当前引脚的端口不一致,或者所在的pull寄存器不一致
+        {
+            *tmp_group_func_addr   = tmp_group_func_data;    //回写功能寄存器
+            *tmp_group_pull_addr   = tmp_group_pull_data;    //回写pull寄存器
+            *tmp_group_dlevel_addr = tmp_group_dlevel_data;  //回写driver level寄存器
+
+            tmp_group_func_addr    = PIO_REG_CFG(port, port_num_func);   //更新功能寄存器地址
+            tmp_group_pull_addr    = PIO_REG_PULL(port, port_num_pull);   //更新pull寄存器
+            tmp_group_dlevel_addr  = PIO_REG_DLEVEL(port, port_num_pull); //更新level寄存器
+
+            tmp_group_func_data    = *tmp_group_func_addr;
+            tmp_group_pull_data    = *tmp_group_pull_addr;
+            tmp_group_dlevel_data  = *tmp_group_dlevel_addr;
+        }
+        else if(pre_port_num_func != port_num_func)                       //如果发现当前引脚的功能寄存器不一致
+        {
+            *tmp_group_func_addr   = tmp_group_func_data;                 //则只回写功能寄存器
+            tmp_group_func_addr    = PIO_REG_CFG(port, port_num_func);   //更新功能寄存器地址
+            tmp_group_func_data    = *tmp_group_func_addr;
+        }
+
+        pre_port_num_pull = port_num_pull;
+        pre_port_num_func = port_num_func;
+        pre_port          = port;
+        //更新功能寄存器
+        tmp_group_func_data &= ~(0x07 << ((port_num - (port_num_func<<3)) << 2));
+        //更新pull状态寄存器
+        tmp_val              =  (port_num - (port_num_pull<<4)) << 1;
+        tmp_group_pull_data &= ~(0x03  << tmp_val);
+        tmp_group_pull_data |= (tmp_sys_gpio_data->hardware_gpio_status.pull & 0x03) << tmp_val;
+        //更新driver状态寄存器
+        tmp_val              =  (port_num - (port_num_pull<<4)) << 1;
+        tmp_group_dlevel_data &= ~(0x03  << tmp_val);
+        tmp_group_dlevel_data |= (tmp_sys_gpio_data->hardware_gpio_status.drv_level & 0x03) << tmp_val;
+    }
+    if(tmp_group_func_addr)                              //只要更新过寄存器地址,就可以对硬件赋值
+    {                                                    //那么把所有的值全部回写到硬件寄存器
+        *tmp_group_func_addr   = tmp_group_func_data;    //回写功能寄存器
+    }
+    if(tmp_group_pull_addr)
+    {
+        *tmp_group_pull_addr   = tmp_group_pull_data;
+    }
+    if(tmp_group_dlevel_addr)
+    {
+        *tmp_group_dlevel_addr = tmp_group_dlevel_data;
+    }
+
+    CSP_OSAL_FREE((char *)p_handler);
+
+    return EGPIO_SUCCESS;
+}
+EXPORT_SYMBOL(gpio_release);
+
+/*
+**********************************************************************************************************************
+*                                               CSP_PIN_Get_All_Gpio_Status
+*
+* Description:
+*                获取用户申请过的所有GPIO的状态
+* Arguments  :
+*        p_handler    :    handler
+*        gpio_status    :    保存用户数据的数组
+*        gpio_count_max    :    数组最大个数,避免数组越界
+*       if_get_user_set_flag   :   读取标志,表示读取用户设定数据或者是实际数据
+* Returns    :
+*
+* Notes      :
+*
+**********************************************************************************************************************
+*/
+__s32  gpio_get_all_pin_status(u32 p_handler, user_gpio_set_t *gpio_status, __u32 gpio_count_max, __u32 if_get_from_hardware)
+{
+    char               *tmp_buf;                                        //转换成char类型
+    __u32               group_count_max, first_port;                    //最大GPIO个数
+    system_gpio_set_t  *user_gpio_set, *tmp_sys_gpio_data;
+    user_gpio_set_t  *script_gpio;
+    __u32               port_num_func, port_num_pull;
+    volatile __u32     *tmp_group_func_addr = NULL, *tmp_group_pull_addr;
+    volatile __u32     *tmp_group_data_addr, *tmp_group_dlevel_addr;
+    __u32               port, port_num;
+    __u32               pre_port = 0x7fffffff, pre_port_num_func = 0x7fffffff, pre_port_num_pull = 0x7fffffff;
+    __u32               i;
+
+    if((!p_handler) || (!gpio_status))
+    {
+        return EGPIO_FAIL;
+    }
+    if(gpio_count_max <= 0)
+    {
+        return EGPIO_FAIL;
+    }
+    tmp_buf = (char *)p_handler;
+    group_count_max = *(int *)tmp_buf;
+    if(group_count_max <= 0)
+    {
+        return EGPIO_FAIL;
+    }
+    user_gpio_set = (system_gpio_set_t *)(tmp_buf + 16);
+    if(group_count_max > gpio_count_max)
+    {
+        group_count_max = gpio_count_max;
+    }
+    //读取用户数据
+    //表示读取用户给定的数据
+    if(!if_get_from_hardware)
+    {
+        for(i = 0; i < group_count_max; i++)
+        {
+            tmp_sys_gpio_data = user_gpio_set + i;             //tmp_sys_gpio_data指向申请的GPIO空间
+            script_gpio       = gpio_status + i;               //script_gpio指向用户传进的空间
+            
+            script_gpio->port      = tmp_sys_gpio_data->port;                       //读出port数据
+            script_gpio->port_num  = tmp_sys_gpio_data->port_num;                   //读出port_num数据
+            script_gpio->pull      = tmp_sys_gpio_data->user_gpio_status.pull;      //读出pull数据
+            script_gpio->mul_sel   = tmp_sys_gpio_data->user_gpio_status.mul_sel;   //读出功能数据
+            script_gpio->drv_level = tmp_sys_gpio_data->user_gpio_status.drv_level; //读出驱动能力数据
+            script_gpio->data      = tmp_sys_gpio_data->user_gpio_status.data;      //读出data数据
+            strcpy(script_gpio->gpio_name, tmp_sys_gpio_data->gpio_name);
+        }
+    }
+    else
+    {
+        for(first_port = 0; first_port < group_count_max; first_port++)
+        {
+            tmp_sys_gpio_data  = user_gpio_set + first_port;
+            port     = tmp_sys_gpio_data->port;               //读出端口数值
+            port_num = tmp_sys_gpio_data->port_num;           //读出端口中的某一个GPIO
+            
+            if(!port)
+            {
+                continue;
+            }
+            port_num_func = (port_num >> 3);
+            port_num_pull = (port_num >> 4);
+            tmp_group_func_addr    = PIO_REG_CFG(port, port_num_func);   //更新功能寄存器地址
+            tmp_group_pull_addr    = PIO_REG_PULL(port, port_num_pull);   //更新pull寄存器
+            tmp_group_dlevel_addr  = PIO_REG_DLEVEL(port, port_num_pull); //更新level寄存器
+            tmp_group_data_addr    = PIO_REG_DATA(port);                  //更新data寄存器
+            break;
+        }
+        if(first_port >= group_count_max)
+        {
+            return 0;
+        }
+        for(i = first_port; i < group_count_max; i++)
+        {
+            tmp_sys_gpio_data = user_gpio_set + i;             //tmp_sys_gpio_data指向申请的GPIO空间
+            script_gpio       = gpio_status + i;               //script_gpio指向用户传进的空间
+
+            port     = tmp_sys_gpio_data->port;                //读出端口数值
+            port_num = tmp_sys_gpio_data->port_num;            //读出端口中的某一个GPIO
+            
+            script_gpio->port = port;                          //读出port数据
+            script_gpio->port_num  = port_num;                 //读出port_num数据
+            strcpy(script_gpio->gpio_name, tmp_sys_gpio_data->gpio_name);
+            
+            port_num_func = (port_num >> 3);
+            port_num_pull = (port_num >> 4);
+            
+            if((port_num_pull != pre_port_num_pull) || (port != pre_port))    //如果发现当前引脚的端口不一致,或者所在的pull寄存器不一致
+            {
+                tmp_group_func_addr    = PIO_REG_CFG(port, port_num_func);   //更新功能寄存器地址
+                tmp_group_pull_addr    = PIO_REG_PULL(port, port_num_pull);   //更新pull寄存器
+                tmp_group_dlevel_addr  = PIO_REG_DLEVEL(port, port_num_pull); //更新level寄存器
+                tmp_group_data_addr    = PIO_REG_DATA(port);                  //更新data寄存器
+            }
+            else if(pre_port_num_func != port_num_func)                       //如果发现当前引脚的功能寄存器不一致
+            {
+                tmp_group_func_addr    = PIO_REG_CFG(port, port_num_func);   //更新功能寄存器地址
+            }
+
+            pre_port_num_pull = port_num_pull;
+            pre_port_num_func = port_num_func;
+            pre_port          = port;
+            //给用户控件赋值
+            script_gpio->pull      = (*tmp_group_pull_addr   >> ((port_num - (port_num_pull<<4))<<1)) & 0x03;    //读出pull数据
+            script_gpio->drv_level = (*tmp_group_dlevel_addr >> ((port_num - (port_num_pull<<4))<<1)) & 0x03;    //读出功能数据
+            script_gpio->mul_sel   = (*tmp_group_func_addr   >> ((port_num - (port_num_func<<3))<<2)) & 0x07;    //读出功能数据
+            if(script_gpio->mul_sel <= 1)
+            {
+                script_gpio->data  = (*tmp_group_data_addr   >>   port_num) & 0x01;                              //读出data数据
+            }
+            else
+            {
+                script_gpio->data = -1;
+            }
+        }
+    }
+
+    return EGPIO_SUCCESS;
+}
+EXPORT_SYMBOL(gpio_get_all_pin_status);
+
+/*
+**********************************************************************************************************************
+*                                               CSP_GPIO_Get_One_PIN_Status
+*
+* Description:
+*                获取用户申请过的所有GPIO的状态
+* Arguments  :
+*        p_handler    :    handler
+*        gpio_status    :    保存用户数据的数组
+*        gpio_name    :    要操作的GPIO的名称
+*       if_get_user_set_flag   :   读取标志,表示读取用户设定数据或者是实际数据
+* Returns    :
+*
+* Notes      :
+*
+**********************************************************************************************************************
+*/
+__s32  gpio_get_one_pin_status(u32 p_handler, user_gpio_set_t *gpio_status, const char *gpio_name, __u32 if_get_from_hardware)
+{
+    char               *tmp_buf;                                        //转换成char类型
+    __u32               group_count_max;                                //最大GPIO个数
+    system_gpio_set_t  *user_gpio_set, *tmp_sys_gpio_data;
+    __u32               port_num_func, port_num_pull;
+    __u32               port, port_num;
+    __u32               i, tmp_val1, tmp_val2;
+    
+    //检查传进的句柄的有效性
+    if((!p_handler) || (!gpio_status))
+    {
+        return EGPIO_FAIL;
+    }
+    tmp_buf = (char *)p_handler;
+    group_count_max = *(int *)tmp_buf;
+    if(group_count_max <= 0)
+    {
+        return EGPIO_FAIL;
+    }
+    else if((group_count_max > 1) && (!gpio_name))
+    {
+        return EGPIO_FAIL;
+    }
+    user_gpio_set = (system_gpio_set_t *)(tmp_buf + 16);
+    //读取用户数据
+    //表示读取用户给定的数据
+    for(i = 0; i < group_count_max; i++)
+    {
+        tmp_sys_gpio_data = user_gpio_set + i;             //tmp_sys_gpio_data指向申请的GPIO空间
+        if(strcmp(gpio_name, tmp_sys_gpio_data->gpio_name))
+        {
+            continue;
+        }
+        strcpy(gpio_status->gpio_name, tmp_sys_gpio_data->gpio_name);
+        port                   = tmp_sys_gpio_data->port;
+        port_num               = tmp_sys_gpio_data->port_num;
+        gpio_status->port      = port;                                              //读出port数据
+        gpio_status->port_num  = port_num;                                          //读出port_num数据
+        
+        if(!if_get_from_hardware)                                                    //当前要求读出用户设计的数据
+        {
+            gpio_status->mul_sel   = tmp_sys_gpio_data->user_gpio_status.mul_sel;   //从用户传进数据中读出功能数据
+            gpio_status->pull      = tmp_sys_gpio_data->user_gpio_status.pull;      //从用户传进数据中读出pull数据
+            gpio_status->drv_level = tmp_sys_gpio_data->user_gpio_status.drv_level; //从用户传进数据中读出驱动能力数据
+            gpio_status->data      = tmp_sys_gpio_data->user_gpio_status.data;      //从用户传进数据中读出data数据
+        }
+        else                                                                        //当前读出寄存器实际的参数
+        {
+        port_num_func = (port_num >> 3);
+        port_num_pull = (port_num >> 4);
+        
+        tmp_val1 = ((port_num - (port_num_func << 3)) << 2);
+        tmp_val2 = ((port_num - (port_num_pull << 4)) << 1);
+        gpio_status->mul_sel   = (PIO_REG_CFG_VALUE(port, port_num_func)>>tmp_val1) & 0x07;       //从硬件中读出功能寄存器
+        gpio_status->pull      = (PIO_REG_PULL_VALUE(port, port_num_pull)>>tmp_val2) & 0x03;      //从硬件中读出pull寄存器
+        gpio_status->drv_level = (PIO_REG_DLEVEL_VALUE(port, port_num_pull)>>tmp_val2) & 0x03;    //从硬件中读出level寄存器
+        if(gpio_status->mul_sel <= 1)
+        {
+            gpio_status->data = (PIO_REG_DATA_VALUE(port) >> port_num) & 0x01;                     //从硬件中读出data寄存器
+        }
+        else
+        {
+            gpio_status->data = -1;
+        }
+        }
+        
+        break;
+    }
+
+    return EGPIO_SUCCESS;
+}
+EXPORT_SYMBOL(gpio_get_one_pin_status);
+
+/*
+**********************************************************************************************************************
+*                                               CSP_PIN_Set_One_Gpio_Status
+*
+* Description:
+*                获取用户申请过的GPIO的某一个的状态
+* Arguments  :
+*        p_handler    :    handler
+*        gpio_status    :    保存用户数据的数组
+*        gpio_name    :    要操作的GPIO的名称
+*       if_get_user_set_flag   :   读取标志,表示读取用户设定数据或者是实际数据
+* Returns    :
+*
+* Notes      :
+*
+**********************************************************************************************************************
+*/
+
+__s32  gpio_set_one_pin_status(u32 p_handler, user_gpio_set_t *gpio_status, const char *gpio_name, __u32 if_set_to_current_input_status)
+{
+    char               *tmp_buf;                                        //转换成char类型
+    __u32               group_count_max;                                //最大GPIO个数
+    system_gpio_set_t  *user_gpio_set, *tmp_sys_gpio_data;
+    user_gpio_set_t     script_gpio;
+    volatile __u32     *tmp_addr;
+    __u32               port_num_func, port_num_pull;
+    __u32               port, port_num;
+    __u32               i, reg_val, tmp_val;
+
+    //检查传进的句柄的有效性
+    if((!p_handler) || (!gpio_name))
+    {
+        return EGPIO_FAIL;
+    }
+    if((if_set_to_current_input_status) && (!gpio_status))
+    {
+        return EGPIO_FAIL;
+    }
+    tmp_buf = (char *)p_handler;
+    group_count_max = *(int *)tmp_buf;
+    if(group_count_max <= 0)
+    {
+        return EGPIO_FAIL;
+    }
+    user_gpio_set = (system_gpio_set_t *)(tmp_buf + 16);
+    //读取用户数据
+    //表示读取用户给定的数据
+    for(i = 0; i < group_count_max; i++)
+    {
+        tmp_sys_gpio_data = user_gpio_set + i;             //tmp_sys_gpio_data指向申请的GPIO空间
+        if(strcmp(gpio_name, tmp_sys_gpio_data->gpio_name))
+        {
+            continue;
+        }
+
+        port          = tmp_sys_gpio_data->port;                           //读出port数据
+        port_num      = tmp_sys_gpio_data->port_num;                       //读出port_num数据
+        port_num_func = (port_num >> 3);
+        port_num_pull = (port_num >> 4);
+
+        if(if_set_to_current_input_status)                                 //根据当前用户设定修正
+        {
+            //修改FUCN寄存器
+            script_gpio.mul_sel   = gpio_status->mul_sel;
+            script_gpio.pull      = gpio_status->pull;
+            script_gpio.drv_level = gpio_status->drv_level;
+            script_gpio.data      = gpio_status->data;
+        }
+        else
+        {
+            script_gpio.mul_sel   = tmp_sys_gpio_data->user_gpio_status.mul_sel;
+            script_gpio.pull      = tmp_sys_gpio_data->user_gpio_status.pull;
+            script_gpio.drv_level = tmp_sys_gpio_data->user_gpio_status.drv_level;
+            script_gpio.data      = tmp_sys_gpio_data->user_gpio_status.data;
+        }
+
+        if(script_gpio.mul_sel >= 0)
+        {
+            tmp_addr = PIO_REG_CFG(port, port_num_func);
+            reg_val = *tmp_addr;                                                       //修改FUNC寄存器
+            tmp_val = (port_num - (port_num_func<<3))<<2;
+            reg_val &= ~(0x07 << tmp_val);
+            reg_val |=  (script_gpio.mul_sel) << tmp_val;
+            *tmp_addr = reg_val;
+        }
+        //修改PULL寄存器
+        if(script_gpio.pull >= 0)
+        {
+            tmp_addr = PIO_REG_PULL(port, port_num_pull);
+            reg_val = *tmp_addr;                                                     //修改FUNC寄存器
+            tmp_val = (port_num - (port_num_pull<<4))<<1;
+            reg_val &= ~(0x03 << tmp_val);
+            reg_val |=  (script_gpio.pull) << tmp_val;
+            *tmp_addr = reg_val;
+        }
+        //修改DLEVEL寄存器
+        if(script_gpio.drv_level >= 0)
+        {
+            tmp_addr = PIO_REG_DLEVEL(port, port_num_pull);
+            reg_val = *tmp_addr;                                                         //修改FUNC寄存器
+            tmp_val = (port_num - (port_num_pull<<4))<<1;
+            reg_val &= ~(0x03 << tmp_val);
+            reg_val |=  (script_gpio.drv_level) << tmp_val;
+            *tmp_addr = reg_val;
+        }
+        //修改data寄存器
+        if(script_gpio.mul_sel == 1)
+        {
+            if(script_gpio.data >= 0)
+            {
+                tmp_addr = PIO_REG_DATA(port);
+                reg_val = *tmp_addr;                                                      //修改DATA寄存器
+                reg_val &= ~(0x01 << port_num);
+                reg_val |=  (script_gpio.data & 0x01) << port_num;
+                *tmp_addr = reg_val;
+            }
+        }
+
+        break;
+    }
+
+    return EGPIO_SUCCESS;
+}
+EXPORT_SYMBOL(gpio_set_one_pin_status);
+
+/*
+**********************************************************************************************************************
+*                                               CSP_GPIO_Set_One_PIN_IO_Status
+*
+* Description:
+*                修改用户申请过的GPIO中的某一个IO口的,输入输出状态
+* Arguments  :
+*        p_handler    :    handler
+*        if_set_to_output_status    :    设置成输出状态还是输入状态
+*        gpio_name    :    要操作的GPIO的名称
+* Returns    :
+*
+* Notes      :
+*
+**********************************************************************************************************************
+*/
+__s32  gpio_set_one_pin_io_status(u32 p_handler, __u32 if_set_to_output_status, const char *gpio_name)
+{
+    char               *tmp_buf;                                        //转换成char类型
+    __u32               group_count_max;                                //最大GPIO个数
+    system_gpio_set_t  *user_gpio_set = NULL, *tmp_sys_gpio_data;
+    volatile __u32      *tmp_group_func_addr = NULL;
+    __u32               port, port_num, port_num_func;
+    __u32                i, reg_val;
+    
+    //检查传进的句柄的有效性
+    if(!p_handler)
+    {
+        return EGPIO_FAIL;
+    }
+    if(if_set_to_output_status > 1)
+    {
+        return EGPIO_FAIL;
+    }
+    tmp_buf = (char *)p_handler;
+    group_count_max = *(int *)tmp_buf;
+    tmp_sys_gpio_data = (system_gpio_set_t *)(tmp_buf + 16);
+    if(group_count_max == 0)
+    {
+        return EGPIO_FAIL;
+    }
+    else if(group_count_max == 1)
+    {
+        user_gpio_set = tmp_sys_gpio_data;
+    }
+    else if(gpio_name)
+    {
+        for(i=0; i<group_count_max; i++)
+        {
+            if(strcmp(gpio_name, tmp_sys_gpio_data->gpio_name))
+            {
+                tmp_sys_gpio_data ++;
+                continue;
+            }
+            user_gpio_set = tmp_sys_gpio_data;
+            break;
+        }
+    }
+    if(!user_gpio_set)
+    {
+        return EGPIO_FAIL;
+    }
+
+    port     = user_gpio_set->port;
+    port_num = user_gpio_set->port_num;
+    port_num_func = port_num >> 3;
+
+    tmp_group_func_addr = PIO_REG_CFG(port, port_num_func);
+    reg_val = *tmp_group_func_addr;
+    reg_val &= ~(0x07 << (((port_num - (port_num_func<<3))<<2)));
+    reg_val |=   if_set_to_output_status << (((port_num - (port_num_func<<3))<<2));
+    *tmp_group_func_addr = reg_val;
+
+    return EGPIO_SUCCESS;
+}
+EXPORT_SYMBOL(gpio_set_one_pin_io_status);
+
+/*
+**********************************************************************************************************************
+*                                               CSP_GPIO_Set_One_PIN_Pull
+*
+* Description:
+*                修改用户申请过的GPIO中的某一个IO口的,PULL状态
+* Arguments  :
+*        p_handler    :    handler
+*        if_set_to_output_status    :    所设置的pull状态
+*        gpio_name    :    要操作的GPIO的名称
+* Returns    :
+*
+* Notes      :
+*
+**********************************************************************************************************************
+*/
+__s32  gpio_set_one_pin_pull(u32 p_handler, __u32 set_pull_status, const char *gpio_name)
+{
+    char               *tmp_buf;                                        //转换成char类型
+    __u32               group_count_max;                                //最大GPIO个数
+    system_gpio_set_t  *user_gpio_set = NULL, *tmp_sys_gpio_data;
+    volatile __u32      *tmp_group_pull_addr = NULL;
+    __u32               port, port_num, port_num_pull;
+    __u32                i, reg_val;
+    //检查传进的句柄的有效性
+    if(!p_handler)
+    {
+        return EGPIO_FAIL;
+    }
+    if(set_pull_status >= 4)
+    {
+        return EGPIO_FAIL;
+    }
+    tmp_buf = (char *)p_handler;
+    group_count_max = *(int *)tmp_buf;
+    tmp_sys_gpio_data = (system_gpio_set_t *)(tmp_buf + 16);
+    if(group_count_max == 0)
+    {
+        return EGPIO_FAIL;
+    }
+    else if(group_count_max == 1)
+    {
+        user_gpio_set = tmp_sys_gpio_data;
+    }
+    else if(gpio_name)
+    {
+        for(i=0; i<group_count_max; i++)
+        {
+            if(strcmp(gpio_name, tmp_sys_gpio_data->gpio_name))
+            {
+                tmp_sys_gpio_data ++;
+                continue;
+            }
+            user_gpio_set = tmp_sys_gpio_data;
+            break;
+        }
+    }
+    if(!user_gpio_set)
+    {
+        return EGPIO_FAIL;
+    }
+
+    port     = user_gpio_set->port;
+    port_num = user_gpio_set->port_num;
+    port_num_pull = port_num >> 4;
+
+    tmp_group_pull_addr = PIO_REG_PULL(port, port_num_pull);
+    reg_val = *tmp_group_pull_addr;
+    reg_val &= ~(0x03 << (((port_num - (port_num_pull<<4))<<1)));
+    reg_val |=  (set_pull_status << (((port_num - (port_num_pull<<4))<<1)));
+    *tmp_group_pull_addr = reg_val;
+
+    return EGPIO_SUCCESS;
+}
+EXPORT_SYMBOL(gpio_set_one_pin_pull);
+
+/*
+**********************************************************************************************************************
+*                                               CSP_GPIO_Set_One_PIN_driver_level
+*
+* Description:
+*                修改用户申请过的GPIO中的某一个IO口的,驱动能力
+* Arguments  :
+*        p_handler    :    handler
+*        if_set_to_output_status    :    所设置的驱动能力等级
+*        gpio_name    :    要操作的GPIO的名称
+* Returns    :
+*
+* Notes      :
+*
+**********************************************************************************************************************
+*/
+__s32  gpio_set_one_pin_driver_level(u32 p_handler, __u32 set_driver_level, const char *gpio_name)
+{
+    char               *tmp_buf;                                        //转换成char类型
+    __u32               group_count_max;                                //最大GPIO个数
+    system_gpio_set_t  *user_gpio_set = NULL, *tmp_sys_gpio_data;
+    volatile __u32      *tmp_group_dlevel_addr = NULL;
+    __u32               port, port_num, port_num_dlevel;
+    __u32                i, reg_val;
+    //检查传进的句柄的有效性
+    if(!p_handler)
+    {
+        return EGPIO_FAIL;
+    }
+    if(set_driver_level >= 4)
+    {
+        return EGPIO_FAIL;
+    }
+    tmp_buf = (char *)p_handler;
+    group_count_max = *(int *)tmp_buf;
+    tmp_sys_gpio_data = (system_gpio_set_t *)(tmp_buf + 16);
+    
+    if(group_count_max == 0)
+    {
+        return EGPIO_FAIL;
+    }
+    else if(group_count_max == 1)
+    {
+        user_gpio_set = tmp_sys_gpio_data;
+    }
+    else if(gpio_name)
+    {
+        for(i=0; i<group_count_max; i++)
+        {
+            if(strcmp(gpio_name, tmp_sys_gpio_data->gpio_name))
+            {
+                tmp_sys_gpio_data ++;
+                continue;
+            }
+            user_gpio_set = tmp_sys_gpio_data;
+            break;
+        }
+    }
+    if(!user_gpio_set)
+    {
+        return EGPIO_FAIL;
+    }
+
+    port     = user_gpio_set->port;
+    port_num = user_gpio_set->port_num;
+    port_num_dlevel = port_num >> 4;
+    
+    tmp_group_dlevel_addr = PIO_REG_DLEVEL(port, port_num_dlevel);
+    reg_val = *tmp_group_dlevel_addr;
+    reg_val &= ~(0x03 << (((port_num - (port_num_dlevel<<4))<<1)));
+    reg_val |=  (set_driver_level << (((port_num - (port_num_dlevel<<4))<<1)));
+    *tmp_group_dlevel_addr = reg_val;
+
+    return EGPIO_SUCCESS;
+}
+EXPORT_SYMBOL(gpio_set_one_pin_driver_level);
+
+/*
+**********************************************************************************************************************
+*                                               CSP_GPIO_Read_One_PIN_Value
+*
+* Description:
+*                读取用户申请过的GPIO中的某一个IO口的端口的电平
+* Arguments  :
+*        p_handler    :    handler
+*        gpio_name    :    要操作的GPIO的名称
+* Returns    :
+*
+* Notes      :
+*
+**********************************************************************************************************************
+*/
+__s32  gpio_read_one_pin_value(u32 p_handler, const char *gpio_name)
+{
+    char               *tmp_buf;                                        //转换成char类型
+    __u32               group_count_max;                                //最大GPIO个数
+    system_gpio_set_t  *user_gpio_set = NULL, *tmp_sys_gpio_data;
+    __u32               port, port_num, port_num_func, func_val;
+    __u32                i, reg_val;
+    //检查传进的句柄的有效性
+    if(!p_handler)
+    {
+        return EGPIO_FAIL;
+    }
+    tmp_buf = (char *)p_handler;
+    group_count_max = *(int *)tmp_buf;
+    tmp_sys_gpio_data = (system_gpio_set_t *)(tmp_buf + 16);
+
+    if(group_count_max == 0)
+    {
+        return EGPIO_FAIL;
+    }
+    else if(group_count_max == 1)
+    {
+        user_gpio_set = tmp_sys_gpio_data;
+    }
+    else if(gpio_name)
+    {
+        for(i=0; i<group_count_max; i++)
+        {
+            if(strcmp(gpio_name, tmp_sys_gpio_data->gpio_name))
+            {
+                tmp_sys_gpio_data ++;
+                continue;
+            }
+            user_gpio_set = tmp_sys_gpio_data;
+            break;
+        }
+    }
+    if(!user_gpio_set)
+    {
+        return EGPIO_FAIL;
+    }
+
+    port     = user_gpio_set->port;
+    port_num = user_gpio_set->port_num;
+    port_num_func = port_num >> 3;
+
+    reg_val  = PIO_REG_CFG_VALUE(port, port_num_func);
+    func_val = (reg_val >> ((port_num - (port_num_func<<3))<<2)) & 0x07;
+    if(func_val == 0)
+    {
+        reg_val = (PIO_REG_DATA_VALUE(port) >> port_num) & 0x01;
+
+        return reg_val;
+    }
+
+    return EGPIO_FAIL;
+}
+EXPORT_SYMBOL(gpio_read_one_pin_value);
+
+/*
+**********************************************************************************************************************
+*                                               CSP_GPIO_Write_One_PIN_Value
+*
+* Description:
+*                修改用户申请过的GPIO中的某一个IO口的端口的电平
+* Arguments  :
+*        p_handler    :    handler
+*       value_to_gpio:  要设置的电平的电压
+*        gpio_name    :    要操作的GPIO的名称
+* Returns    :
+*
+* Notes      :
+*
+**********************************************************************************************************************
+*/
+__s32  gpio_write_one_pin_value(u32 p_handler, __u32 value_to_gpio, const char *gpio_name)
+{
+    char               *tmp_buf;                                        //转换成char类型
+    __u32               group_count_max;                                //最大GPIO个数
+    system_gpio_set_t  *user_gpio_set = NULL, *tmp_sys_gpio_data;
+    volatile __u32     *tmp_group_data_addr = NULL;
+    __u32               port, port_num, port_num_func, func_val;
+    __u32                i, reg_val;
+    //检查传进的句柄的有效性
+    if(!p_handler)
+    {
+        return EGPIO_FAIL;
+    }
+    if(value_to_gpio >= 2)
+    {
+        return EGPIO_FAIL;
+    }
+    tmp_buf = (char *)p_handler;
+    group_count_max = *(int *)tmp_buf;
+    tmp_sys_gpio_data = (system_gpio_set_t *)(tmp_buf + 16);
+    
+    if(group_count_max == 0)
+    {
+        return EGPIO_FAIL;
+    }
+    else if(group_count_max == 1)
+    {
+        user_gpio_set = tmp_sys_gpio_data;
+    }
+    else if(gpio_name)
+    {
+        for(i=0; i<group_count_max; i++)
+        {
+            if(strcmp(gpio_name, tmp_sys_gpio_data->gpio_name))
+            {
+                tmp_sys_gpio_data ++;
+                continue;
+            }
+            user_gpio_set = tmp_sys_gpio_data;
+            break;
+        }
+    }
+    if(!user_gpio_set)
+    {
+        return EGPIO_FAIL;
+    }
+
+    port     = user_gpio_set->port;
+    port_num = user_gpio_set->port_num;
+    port_num_func = port_num >> 3;
+
+    reg_val  = PIO_REG_CFG_VALUE(port, port_num_func);
+    func_val = (reg_val >> ((port_num - (port_num_func<<3))<<2)) & 0x07;
+    if(func_val == 1)
+    {
+        tmp_group_data_addr = PIO_REG_DATA(port);
+        reg_val = *tmp_group_data_addr;
+        reg_val &= ~(1 << port_num);
+        reg_val |=  (value_to_gpio << port_num);
+        *tmp_group_data_addr = reg_val;
+
+        return EGPIO_SUCCESS;
+    }
+
+    return EGPIO_FAIL;
+}
+EXPORT_SYMBOL(gpio_write_one_pin_value);
+
diff --git a/lichee/linux-3.3/drivers/ata/ahci.h b/lichee/linux-3.3/drivers/ata/ahci.h
index b175000..c2594dd 100644
--- a/lichee/linux-3.3/drivers/ata/ahci.h
+++ b/lichee/linux-3.3/drivers/ata/ahci.h
@@ -195,6 +195,9 @@ enum {
        PORT_FBS_EN                = (1 << 0), /* Enable FBS */

        /* hpriv->flags bits */
+
+#define AHCI_HFLAGS(flags)                .private_data        = (void *)(flags)
+
        AHCI_HFLAG_NO_NCQ                = (1 << 0),
        AHCI_HFLAG_IGN_IRQ_IF_ERR        = (1 << 1), /* ignore IRQ_IF_ERR */
        AHCI_HFLAG_IGN_SERR_INTERNAL        = (1 << 2), /* ignore SERR_INTERNAL */
@@ -210,6 +213,9 @@ enum {
        AHCI_HFLAG_NO_SNTF                = (1 << 12), /* no sntf */
        AHCI_HFLAG_NO_FPDMA_AA                = (1 << 13), /* no FPDMA AA */
        AHCI_HFLAG_YES_FBS                = (1 << 14), /* force FBS cap on */
+        AHCI_HFLAG_DELAY_ENGINE                = (1 << 15), /* do not start engine on
+                                                        port start (wait until
+                                                        error-handling stage) */

        /* ap->flags bits */

diff --git a/lichee/linux-3.3/drivers/ata/libahci.c b/lichee/linux-3.3/drivers/ata/libahci.c
index 6d083e0..3591865 100644
--- a/lichee/linux-3.3/drivers/ata/libahci.c
+++ b/lichee/linux-3.3/drivers/ata/libahci.c
@@ -46,6 +46,8 @@
#include <linux/libata.h>
#include "ahci.h"

+#include <linux/slab.h>
+
static int ahci_skip_host_reset;
int ahci_ignore_sss;
EXPORT_SYMBOL_GPL(ahci_ignore_sss);
@@ -744,6 +746,7 @@ static void ahci_power_down(struct ata_port *ap)

static void ahci_start_port(struct ata_port *ap)
{
+        struct ahci_host_priv *hpriv = ap->host->private_data;
        struct ahci_port_priv *pp = ap->private_data;
        struct ata_link *link;
        struct ahci_em_priv *emp;
@@ -753,6 +756,10 @@ static void ahci_start_port(struct ata_port *ap)
        /* enable FIS reception */
        ahci_start_fis_rx(ap);

+        /* enable DMA */
+        if (!(hpriv->flags & AHCI_HFLAG_DELAY_ENGINE))
+                ahci_start_engine(ap);
+
        /* turn on LEDs */
        if (ap->flags & ATA_FLAG_EM) {
                ata_for_each_link(link, ap, EDGE) {
@@ -2069,11 +2076,7 @@ static int ahci_port_start(struct ata_port *ap)
                rx_fis_sz = AHCI_RX_FIS_SZ;
        }

-        //mem = dmam_alloc_coherent(dev, dma_sz, &mem_dma, GFP_KERNEL);
-        //danielwang
-        mem = kmalloc(dma_sz, GFP_DMA | GFP_KERNEL);
-        mem_dma = __pa(mem);
-        //printk("dmam_alloc_coherent mem = 0x%x, size = 0x%x, mem_dma=0x%x\n", (unsigned int)mem, (unsigned int)dma_sz, (unsigned int)mem_dma);  
+        mem = dmam_alloc_coherent(dev, dma_sz, &mem_dma, GFP_KERNEL);
        if (!mem)
                return -ENOMEM;
        memset(mem, 0, dma_sz);
diff --git a/lichee/linux-3.3/drivers/ata/sw_ahci_platform.c b/lichee/linux-3.3/drivers/ata/sw_ahci_platform.c
index 1cc2206..35dbb45 100755
--- a/lichee/linux-3.3/drivers/ata/sw_ahci_platform.c
+++ b/lichee/linux-3.3/drivers/ata/sw_ahci_platform.c
@@ -1,25 +1,31 @@
/*
-*************************************************************************************
-*                                                       Linux
-*                                                   AHCI SATA platform driver
-*
-*                                        (c) Copyright 2006-2010, All winners Co,Ld.
-*                                                               All Rights Reserved
-*
-* File Name         : sw_ahci_platform.c
-*
-* Author                 : danielwang
-*
-* Description         : SATA Host Controller Driver for AW1623 Platform
-*
-* Notes         : 
-*
-* History                 :
-*      <author>                    <time>               <version >                    <desc>
-*    danielwang        2011-6-29            1.0          create this file
-*
-*************************************************************************************
-*/
+ * drivers/ata/sw_ahci_platform.c
+ *
+ * (C) Copyright 2007-2012
+ * Allwinner Technology Co., Ltd. <www.allwinnertech.com>
+ * Daniel Wang <[email protected]>
+ *
+ * Based on ahci_platform.c AHCI SATA platform driver
+ *
+ * Copyright 2004-2005  Red Hat, Inc.
+ *   Jeff Garzik <[email protected]>
+ * Copyright 2010  MontaVista Software, LLC.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of
+ * the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.         See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
+ * MA 02111-1307 USA
+ */

#include <linux/kernel.h>
#include <linux/gfp.h>
@@ -34,7 +40,6 @@

#include <linux/clk.h>
#include <mach/sys_config.h>
-#include <mach/gpio.h>
#include "sw_ahci_platform.h"

static struct scsi_host_template ahci_platform_sht = {
@@ -66,89 +71,89 @@ static int sw_ahci_phy_init(unsigned int base)
        unsigned int tmp;
        const unsigned int timeout_val = 0x100000;
        unsigned int timeout = timeout_val;
-        
+
        for(tmp=0; tmp<0x1000; tmp++);
-        
+
        SW_AHCI_ACCESS_LOCK(base, 0);
-        
+
        tmp = ahci_readl(base, SW_AHCI_PHYCS1R_OFFSET);
-        tmp |= (0x1<<19); 
+        tmp |= (0x1<<19);
        ahci_writel(base, SW_AHCI_PHYCS1R_OFFSET, tmp);
-                
+
        tmp = ahci_readl(base, SW_AHCI_PHYCS0R_OFFSET);
-        tmp |= 0x1<<23; 
-        tmp |= 0x1<<18; 
-        tmp &= ~(0x7<<24); 
-        tmp |= 0x5<<24; 
+        tmp |= 0x1<<23;
+        tmp |= 0x1<<18;
+        tmp &= ~(0x7<<24);
+        tmp |= 0x5<<24;
        ahci_writel(base, SW_AHCI_PHYCS0R_OFFSET, tmp);
-        
+
        tmp = ahci_readl(base, SW_AHCI_PHYCS1R_OFFSET);
-        tmp &= ~(0x3<<16); 
-        tmp |= (0x2<<16);         
+        tmp &= ~(0x3<<16);
+        tmp |= (0x2<<16);
        tmp &= ~(0x1f<<8);
-        tmp |= (6<<8);         
-        tmp &= ~(0x3<<6); 
-        tmp |= (2<<6);         
+        tmp |= (6<<8);
+        tmp &= ~(0x3<<6);
+        tmp |= (2<<6);
        ahci_writel(base, SW_AHCI_PHYCS1R_OFFSET, tmp);
-        
+
        tmp = ahci_readl(base, SW_AHCI_PHYCS1R_OFFSET);
-        tmp |= (0x1<<28); 
-        tmp |= (0x1<<15); 
+        tmp |= (0x1<<28);
+        tmp |= (0x1<<15);
        ahci_writel(base, SW_AHCI_PHYCS1R_OFFSET, tmp);
-        
+
        tmp = ahci_readl(base, SW_AHCI_PHYCS1R_OFFSET);
-        tmp &= ~(0x1<<19); 
+        tmp &= ~(0x1<<19);
        ahci_writel(base, SW_AHCI_PHYCS1R_OFFSET, tmp);
-                
+
        tmp = ahci_readl(base, SW_AHCI_PHYCS0R_OFFSET);
-        tmp &= ~(0x7<<20); 
-        tmp |= (0x03<<20);  
+        tmp &= ~(0x7<<20);
+        tmp |= (0x03<<20);
        ahci_writel(base, SW_AHCI_PHYCS0R_OFFSET, tmp);
-                
+
        tmp = ahci_readl(base, SW_AHCI_PHYCS2R_OFFSET);
-        tmp &= ~(0x1f<<5); 
-        tmp |= (0x19<<5);  
+        tmp &= ~(0x1f<<5);
+        tmp |= (0x19<<5);
        ahci_writel(base, SW_AHCI_PHYCS2R_OFFSET, tmp);
-                
+
        for(tmp=0; tmp<0x1000; tmp++);
-        
+
        tmp = ahci_readl(base, SW_AHCI_PHYCS0R_OFFSET);
-        tmp |= 0x1<<19; 
+        tmp |= 0x1<<19;
        ahci_writel(base, SW_AHCI_PHYCS0R_OFFSET, tmp);
-                
+
        timeout = timeout_val;
        do{
                tmp = ahci_readl(base, SW_AHCI_PHYCS0R_OFFSET);
                timeout --;
                if(!timeout) break;
-        }while((tmp&(0x7<<28))!=(0x02<<28)); 
-        
+        }while((tmp&(0x7<<28))!=(0x02<<28));
+
        if(!timeout)
        {
                printk("SATA AHCI Phy Power Failed!!\n");
        }
-        
+
        tmp = ahci_readl(base, SW_AHCI_PHYCS2R_OFFSET);
-        tmp |= 0x1<<24; 
+        tmp |= 0x1<<24;
        ahci_writel(base, SW_AHCI_PHYCS2R_OFFSET, tmp);
-        
+
        timeout = timeout_val;
        do{
                tmp = ahci_readl(base, SW_AHCI_PHYCS2R_OFFSET);
                timeout --;
                if(!timeout) break;
-        }while(tmp&(0x1<<24)); 
-        
+        }while(tmp&(0x1<<24));
+
        if(!timeout)
        {
                printk("SATA AHCI Phy Calibration Failed!!\n");
        }
-        
+
        for(tmp=0; tmp<0x3000; tmp++);
-        
+
        SW_AHCI_ACCESS_LOCK(base, 0x07);
-        
-        return 0;                
+
+        return 0;
}

static int sw_ahci_start(struct device *dev, void __iomem *addr)
@@ -156,16 +161,7 @@ static int sw_ahci_start(struct device *dev, void __iomem *addr)
        struct clk *hclk;
        struct clk *mclk;
        u32 pio_hdle = 0;
-        int ctrl = 0;
        int rc = 0;
-                
-        script_parser_fetch(sw_ahci_para_name, sw_ahci_used_name, &ctrl, sizeof(int));
-        if(!ctrl)
-        {
-                dev_err(dev, "AHCI is disable\n");
-                rc = -EINVAL;
-            goto err2;
-        }        

        /*Enable mclk and hclk for AHCI*/
        mclk = clk_get(dev, sw_ahci_mclk_name);
@@ -175,7 +171,7 @@ static int sw_ahci_start(struct device *dev, void __iomem *addr)
             rc = -EINVAL;
             goto err2;
     }
-    
+
        hclk = clk_get(dev, sw_ahci_hclk_name);
        if (IS_ERR(hclk))
        {
@@ -183,25 +179,25 @@ static int sw_ahci_start(struct device *dev, void __iomem *addr)
             rc = -EINVAL;
             goto err1;
        }
-        
+
        /*Enable SATA Clock in SATA PLL*/
        ahci_writel(CCMU_PLL6_VBASE, 0, ahci_readl(CCMU_PLL6_VBASE, 0)|(0x1<<14));
        clk_enable(mclk);
        clk_enable(hclk);
-        
+
        sw_ahci_phy_init((unsigned int)addr);

-        pio_hdle = sw_gpio_request_ex(sw_ahci_para_name, NULL);
-        if(pio_hdle) 
+        pio_hdle = gpio_request_ex(sw_ahci_para_name, NULL);
+        if(pio_hdle)
        {
-                sw_gpio_write_one_pin_value(pio_hdle, 1, sw_ahci_gpio_name);        
-                sw_gpio_release(pio_hdle, 2);
+                gpio_write_one_pin_value(pio_hdle, 1, sw_ahci_gpio_name);
+                gpio_release(pio_hdle, 2);
        }
-        
+
        clk_put(hclk);
err1:
-        clk_put(mclk);        
-err2:        
+        clk_put(mclk);
+err2:
        return rc;
}

@@ -211,7 +207,7 @@ static void sw_ahci_stop(struct device *dev)
        struct clk *mclk;
        u32 pio_hdle = 0;
        int rc = 0;
-                
+
        mclk = clk_get(dev, sw_ahci_mclk_name);
        if (IS_ERR(mclk))
     {
@@ -219,7 +215,7 @@ static void sw_ahci_stop(struct device *dev)
             rc = -EINVAL;
             goto err2;
     }
-    
+
        hclk = clk_get(dev, sw_ahci_hclk_name);
        if (IS_ERR(hclk))
        {
@@ -228,13 +224,13 @@ static void sw_ahci_stop(struct device *dev)
             goto err1;
        }

-        pio_hdle = sw_gpio_request_ex(sw_ahci_para_name, NULL);
-        if(pio_hdle) 
+        pio_hdle = gpio_request_ex(sw_ahci_para_name, NULL);
+        if(pio_hdle)
        {
-                sw_gpio_write_one_pin_value(pio_hdle, 0, sw_ahci_gpio_name);
-                sw_gpio_release(pio_hdle, 2);
+                gpio_write_one_pin_value(pio_hdle, 0, sw_ahci_gpio_name);
+                gpio_release(pio_hdle, 2);
        }
-        
+
        /*Disable mclk and hclk for AHCI*/
        clk_disable(mclk);
        clk_disable(hclk);
@@ -252,7 +248,7 @@ static struct ata_port_info sw_ahci_port_info = {
        //.mwdma_mask = ,
        .udma_mask = ATA_UDMA6,
        .port_ops = &ahci_ops,
-        .private_data = (void*)(AHCI_HFLAG_32BIT_ONLY | AHCI_HFLAG_NO_MSI 
+        .private_data = (void*)(AHCI_HFLAG_32BIT_ONLY | AHCI_HFLAG_NO_MSI
                                                        | AHCI_HFLAG_NO_PMP | AHCI_HFLAG_YES_NCQ),
};

@@ -262,13 +258,20 @@ static struct ahci_platform_data sw_ahci_platform_data = {
        .exit = sw_ahci_stop,
};

+void sw_ahci_release(struct device *dev)
+{
+        /* FILL ME! */
+}
+
static struct platform_device sw_ahci_device = {
        .name                = "sw_ahci",
        .id                        = 0,
        .dev                 = {
                .platform_data = &sw_ahci_platform_data,
+                .release = &sw_ahci_release,
+                .coherent_dma_mask = DMA_BIT_MASK(32),
        },
-        
+
        .num_resources        = ARRAY_SIZE(sw_ahci_resources),
        .resource                = sw_ahci_resources,
};
@@ -421,7 +424,7 @@ static int __devexit sw_ahci_remove(struct platform_device *pdev)
void sw_ahci_dump_reg(struct device *dev)
{
        struct ata_host *host = dev_get_drvdata(dev);
-        struct ahci_host_priv *hpriv = host->private_data; 
+        struct ahci_host_priv *hpriv = host->private_data;
        u32 base = (u32)hpriv->mmio;
        int i = 0;

@@ -434,7 +437,7 @@ void sw_ahci_dump_reg(struct device *dev)

static int sw_ahci_suspend(struct device *dev)
{
-        
+
        printk("sw_ahci_platform: sw_ahci_suspend\n"); //danielwang
        //sw_ahci_dump_reg(dev);

@@ -447,12 +450,12 @@ extern int ahci_hardware_recover_for_controller_resume(struct ata_host *host);
static int sw_ahci_resume(struct device *dev)
{
        struct ata_host *host = dev_get_drvdata(dev);
-        struct ahci_host_priv *hpriv = host->private_data; 
-                
+        struct ahci_host_priv *hpriv = host->private_data;
+
        printk("sw_ahci_platform: sw_ahci_resume\n"); //danielwang

        sw_ahci_start(dev, hpriv->mmio);
-        //sw_ahci_dump_reg(dev);        
+        //sw_ahci_dump_reg(dev);

        ahci_hardware_recover_for_controller_resume(host);

@@ -483,14 +486,30 @@ static struct platform_driver sw_ahci_driver = {

static int __init sw_ahci_init(void)
{
-        platform_device_register(&sw_ahci_device);        
-        return platform_driver_probe(&sw_ahci_driver, sw_ahci_probe);
+        int rc, ctrl = 0;
+        script_parser_fetch(sw_ahci_para_name,
+                        sw_ahci_used_name, &ctrl, sizeof(int));
+        if (!ctrl) {
+                pr_warn("AHCI is disabled in script.bin\n");
+                return -ENODEV;
+        }
+
+        rc = platform_device_register(&sw_ahci_device);
+        if (rc)
+                return rc;
+
+        rc = platform_driver_probe(&sw_ahci_driver, sw_ahci_probe);
+        if (rc)
+                platform_device_unregister(&sw_ahci_device);
+
+        return rc;
}
module_init(sw_ahci_init);

static void __exit sw_ahci_exit(void)
{
        platform_driver_unregister(&sw_ahci_driver);
+        platform_device_unregister(&sw_ahci_device);
}
module_exit(sw_ahci_exit);

diff --git a/lichee/linux-3.3/drivers/ata/sw_ahci_platform.h b/lichee/linux-3.3/drivers/ata/sw_ahci_platform.h
index 45abd76..fb35f5b 100755
--- a/lichee/linux-3.3/drivers/ata/sw_ahci_platform.h
+++ b/lichee/linux-3.3/drivers/ata/sw_ahci_platform.h
@@ -1,20 +1,38 @@
/*
- * AHCI SATA platform driver
+ * drivers/ata/sw_ahci_platform.h
  *
+ * (C) Copyright 2007-2012
+ * Allwinner Technology Co., Ltd. <www.allwinnertech.com>
+ * Daniel Wang <[email protected]>
+ *
+ * Based on ahci_platform.h
+ * 
  * Copyright 2004-2005  Red Hat, Inc.
  *   Jeff Garzik <[email protected]>
  * Copyright 2010  MontaVista Software, LLC.
  *   Anton Vorontsov <[email protected]>
  *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of
+ * the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.         See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
+ * MA 02111-1307 USA
  */

#ifndef _SW_AHCI_PLATFORM_H
#define _SW_AHCI_PLATFORM_H

+#include <mach/irqs.h>
+
#define        ahci_readb(base,offset)                  (*((volatile unsigned char*)((base)+(offset))))
#define        ahci_readw(base,offset)                  (*((volatile unsigned short*)((base)+(offset))))
#define        ahci_readl(base,offset)                  (*((volatile unsigned int*)((base)+(offset))))
@@ -48,9 +66,9 @@
#define SW_AHCI_P0PHYCR_OFFSET                0x0178
#define SW_AHCI_P0PHYSR_OFFSET                0x017C

-#define SW_AHCI_ACCESS_LOCK(base,x)                (*((volatile unsigned int *)((base)+SW_AHCI_RWCR_OFFSET)) = (x))  
+#define SW_AHCI_ACCESS_LOCK(base,x)                (*((volatile unsigned int *)((base)+SW_AHCI_RWCR_OFFSET)) = (x))

-#define INTC_IRQNO_AHCI                                56
+#define INTC_IRQNO_AHCI                                AW_IRQ_SATA

#define CCMU_PLL6_VBASE                                0xF1C20028

diff --git a/lichee/tools/pack/chips/sun7i/configs/android/wing-k70/sys_config.fex b/lichee/tools/pack/chips/sun7i/configs/android/wing-k70/sys_config.fex
index e50d6e6..15eb781 100755
--- a/lichee/tools/pack/chips/sun7i/configs/android/wing-k70/sys_config.fex
+++ b/lichee/tools/pack/chips/sun7i/configs/android/wing-k70/sys_config.fex
@@ -762,7 +762,7 @@ tvin_channel_num    = 4
;-------------------------------------------------------------------------------
[sata_para]
sata_used           = 1
-sata_power_en       =
+sata_power_en       = portH17<1><default><default><1>


;-------------------------------------------------------------------------------
-- 

1.7.9.5

格式化硬盘脚本文件:

#! /system/bin/sh

################################################################################

#Title: fdisk_shell.sh

#Description: fdisk disk

#Platform:A20 

#Copyright: Copyright (c) 2013

#Company:< [email protected]>

#Author: wangjian

#Version: 1.0

#Date: 2013.11.18

/system/bin/busybox mkdir -p /mnt/sdcard/fdisk

/system/bin/busybox chmod 777 /mnt/sdcard/fdisk

/system/bin/busybox touch /mnt/sdcard/fdisk/2

/system/bin/busybox touch /mnt/sdcard/fdisk/1.0

/system/bin/busybox touch /mnt/sdcard/fdisk/1.1

/system/bin/busybox touch /mnt/sdcard/fdisk/1.2

/system/bin/ll /mnt/sata/

umount /mnt/sata

sleep 1

#########################fdisk operation########################################

echo "n

p

1

1


t

L

7

w" >/mnt/sdcard/fdisk/2

###########################fdisk disk###########################################

/system/bin/busybox fdisk /dev/block/vold/sata/8:0 -l | /system/bin/busybox grep dev > /mnt/sdcard/fdisk/1.0

/system/bin/busybox fdisk /dev/block/vold/sata/8:0 -l | /system/bin/busybox grep dev | /system/bin/busybox sed -n '1,/(LBA)/p' | /system/bin/busybox sed '1d' | /system/bin/busybox awk '{print $1}' | /system/bin/busybox sed 's#/dev/block/vold/sata/8:0p##g' >  /mnt/sdcard/fdisk/1.1

/system/bin/busybox fdisk /dev/block/vold/sata/8:0 -l | /system/bin/busybox grep dev | /system/bin/busybox sed -n '/(LBA)/,$p' | /system/bin/busybox sed '1d' | /system/bin/busybox awk '{print $1}' | /system/bin/busybox sed 's#/dev/block/vold/sata/8:0p##g' > /mnt/sdcard/fdisk/1.2


echo "/mnt/sdcard/fdisk/1.0" 

cat /mnt/sdcard/fdisk/1.0

echo "/mnt/sdcard/fdisk/1.1"

cat /mnt/sdcard/fdisk/1.1

echo "/mnt/sdcard/fdisk/1.2"

cat /mnt/sdcard/fdisk/1.2


y=`cat /mnt/sdcard/fdisk/1.1 | /system/bin/busybox wc -l`

echo $y

i=1


while [[ "$i" -lt "$y" ]];do

j=`/system/bin/busybox sed -n "${i}p" /mnt/sdcard/fdisk/1.1` 

echo "d

${j}

w" >/mnt/sdcard/fdisk/2_"${i}"

echo "/mnt/sdcard/fdisk/2_"${i}""

umount /mnt/sata/8_"${j}"

cat /mnt/sdcard/fdisk/2_"${i}"

echo "delete the "${j}" partition....."

/system/bin/busybox fdisk /dev/block/vold/sata/8:0 < /mnt/sdcard/fdisk/2_"${i}"

let i=i+1

done

############

if [[ -z "`grep "(LBA)" /mnt/sdcard/fdisk/1.0`" ]];then

j=`/system/bin/busybox sed -n "${i}p" /mnt/sdcard/fdisk/1.1` 

echo "d

w" >/mnt/sdcard/fdisk/2_"${i}"

fi

#############

if [[ -n "`grep "(LBA)" /mnt/sdcard/fdisk/1.0`" ]];then

j=`/system/bin/busybox sed -n "${i}p" /mnt/sdcard/fdisk/1.1` 

echo "d

${j}

w" >/mnt/sdcard/fdisk/2_"${i}"

fi

########delete the last partition#########################

if [[ "$y" -ne 0 ]];then

echo "/mnt/sdcard/fdisk/2_"${i}""

umount /mnt/sata/8_"${j}"

cat /mnt/sdcard/fdisk/2_"${i}"

echo "delete the last partition....."

/system/bin/busybox fdisk /dev/block/vold/sata/8:0 < /mnt/sdcard/fdisk/2_"${i}"

fi

#####################umount the least partition#############

if [[ -n "`grep "(LBA)" /mnt/sdcard/fdisk/1.0`" ]];then

y=`cat /mnt/sdcard/fdisk/1.2 | /system/bin/busybox wc -l`

i=1

while [[ "$i" -le "$y" ]];do

j=`/system/bin/busybox sed -n "${i}p" /mnt/sdcard/fdisk/1.2`

echo "umount /mnt/sata/8_"${j}""

umount /mnt/sata/8_"${j}"

let i=i+1  

done

fi

#################mkfs######################################

echo "fdisk....." 

/system/bin/busybox fdisk /dev/block/vold/sata/8:0  < /mnt/sdcard/fdisk/2

echo "umount /mnt/sata."

umount /mnt/sata

sleep 1

echo "mkfs....."

mkntfs -f -L sata /dev/block/vold/sata/8:1

echo "fs has done!"

echo "mount...."

#mount -t ntfs /dev/block/vold/sata/8:1 /mnt/sata

reboot




你可能感兴趣的:(A20sata热拔插的实现)