iTop4412 uboot-2019.2移植之内存控制器(七)

一、说明

内存控制器非常难配置,故而不能出错。我严格按照文档上的步骤配置,其中的参数自行体会。

二、配置流程

首先配置内存交错,然后初始化每个DMC。

223 void mem_ctrl_init(int reset)
224 {
225     struct exynos4_dmc *dmc1 = (struct exynos4_dmc *)samsung_get_base_dmc_ctrl();
226     struct exynos4_dmc *dmc2 = (struct exynos4_dmc *)(samsung_get_base_dmc_ctrl() + DMC_OFFSET);
227     /**
228     ¦* 配置内存交错
229     ¦*/
230     writel(APB_SFR_INTERLEAVE_CONF_VAL, &dmc1->ivcontrol);
231     writel(APB_SFR_INTERLEAVE_CONF_VAL, &dmc2->ivcontrol);
232
233     /*
234     ¦* 初始化内存控制器
235     ¦*/
236     dmc_init(dmc1);
237     dmc_init(dmc2);
238 }

配置步骤在代码中,对比手册阅读

static void dmc_init(struct exynos4_dmc *dmc)
{
    /**
     * Setup 2:
     *  enable PhyControl1.term_write_en, PhyControl1.term_read_en
     */
    writel(mem.control1, &dmc->phycontrol1); //有差异

    /**
     * Stup 3:
     *  disable PhyZQControl.ctrl_zq_mode_noterm
     *  enable PhyZQControl.ctrl_zq_start
     */
    writel(mem.zqcontrol, &dmc->phyzqcontrol);


    /**
     * Setup 4:
     *  set PhyControl0.ctrl_start_point, PhyControl0.ctrl_inc
     *  set PhyControl0.ctrl_dll_on to 1 ——activate PHY DLL
     */
    phy_control_reset(0, dmc);

    /**
     * Setup 5:
     *  set PhyControl1.ctr_shiftc, PhyControl1.ctrl_offsetc
     */
    writel(mem.control1, &dmc->phycontrol1);

    /**
     * Setup 6:
     *  set PhyControl0.ctrl_start to 1
     */
    writel((mem.control0 | CTRL_START | CTRL_DLL_ON), &dmc->phycontrol0);


    /**
     * Setup 7:
     *  set ConControl, close auto refresh
     */
    writel(mem.concontrol, &dmc->concontrol);


    /**
     * Setup 8:
     *  set MemControl, close power down modes, close pzq_en
     */
    writel(mem.memcontrol, &dmc->memcontrol); //差异

    /**
     * Setup 9:
     *  set Memory info
     */
    writel(mem.memconfig0, &dmc->memconfig0);
    writel(mem.memconfig1, &dmc->memconfig1);

    /**
     * Setup 10:
     *  set PrechConfig
     */
    writel(mem.prechconfig, &dmc->prechconfig);

    /**
     * Setup 11:
     *  set TimingAref, TimingRow, TimingData and TimingPower
     */
    writel(mem.timingref, &dmc->timingref);
    writel(mem.timingrow, &dmc->timingrow);
    writel(mem.timingdata, &dmc->timingdata);
    writel(mem.timingpower, &dmc->timingpower);

    /**
     * Setup 13:
     *  wait PhyStatus0.ctrl_clock and PhyStatus0.ctrl_flock to 1
     */

    while(!(dmc->phystatus & 2));

    /**
     * Setup 15,16:
     *  set PhyContron1.fp_resync to 1
     */
    phy_control_reset(1, dmc);

    /**
     * Setup 19:
     *  NOP command
     *  hold CKE to logic high level
     *  chip 0
     */
    writel(DIRECT_CMD_NOP, &dmc->directcmd);

    /**
     * Setup 21:
     *  send EMRS2 command
     *  send EMRS3 command
     *  send EMRS command
     *  send MRS command
     *  chip 0
     */
    dmc_config_mrs(dmc, 0);

    /**
     *Setup 26:
     send ZQINIT command
     chip 0
     */
    writel(DIRECT_CMD_ZQ, &dmc->directcmd);

    /**
     * Setup 19:
     *  NOP command
     *  hold CKE to logic high level
     *  chip 1
     */
    writel((DIRECT_CMD_NOP | DIRECT_CMD_CHIP1_SHIFT), &dmc->directcmd);

    /**
     * Setup 21:
     *  send EMRS2 command
     *  send EMRS3 command
     *  send EMRS command
     *  send MRS command
     *  chip 1
     */
    dmc_config_mrs(dmc, 1);

    /**
     *Setup 26:
     send ZQINIT command
     chip 1
     */
    writel((DIRECT_CMD_ZQ | DIRECT_CMD_CHIP1_SHIFT), &dmc->directcmd);

    /**
     * Setup 28:
     *  set ConControl auto refresh
     */
    writel((mem.concontrol | AREF_EN), &dmc->concontrol);

    /**
     * Setup 29:
     *  set MemControl
     */
    writel((mem.memcontrol | MEMCONTROL_OR), &dmc->memcontrol);
}

三、配置参数

/********************************************************************************
 *
 * 内存控制器的配置
 *
 ********************************************************************************/

/* DMC */
#define DIRECT_CMD_CHIP1_SHIFT  (1 << 20)
#define MEM_TIMINGS_MSR_COUNT   4
#define CTRL_START  (1 << 0)
#define CTRL_DLL_ON (1 << 1)
#define AREF_EN     (1 << 5)
#define DRV_TYPE    (1 << 6)

struct mem_timings {
    unsigned direct_cmd_msr[MEM_TIMINGS_MSR_COUNT];
    unsigned timingref;
    unsigned timingrow;
    unsigned timingdata;
    unsigned timingpower;
    unsigned zqcontrol;
    unsigned control0;
    unsigned control1;
    unsigned control2;
    unsigned concontrol;
    unsigned prechconfig;
    unsigned memcontrol;
    unsigned memconfig0;
    unsigned memconfig1;
    unsigned dll_resync;
    unsigned dll_on;
};

/* MIU */
/* MIU Config Register Offsets*/

#ifdef CONFIG_ITOP4412
#define APB_SFR_INTERLEAVE_CONF_VAL     0x80000007
#endif

#ifdef CONFIG_MIU_1BIT_INTERLEAVED
#define APB_SFR_INTERLEAVE_CONF_VAL 0x0000000c
#endif

#ifdef CONFIG_MIU_2BIT_INTERLEAVED
#define APB_SFR_INTERLEAVE_CONF_VAL 0x2000150c
#endif


#define FORCE_DLL_RESYNC    3
#define DLL_CONTROL_ON      1

#define DIRECT_CMD_NOP  0x07000000
#define DIRECT_CMD1     0x00020000
#define DIRECT_CMD2     0x00030000
#define DIRECT_CMD3     0x00010002
#define DIRECT_CMD4     0x00000328
#define DIRECT_CMD_ZQ   0x0a000000

#define CTRL_ZQ_MODE_NOTERM (0x1 << 0)
#define CTRL_ZQ_START       (0x1 << 1)
#define CTRL_ZQ_DIV         (0x0 << 4)
#define CTRL_ZQ_MODE_DDS    (0x7 << 8)
#define CTRL_ZQ_MODE_TERM   (0x2 << 11)
#define CTRL_ZQ_FORCE_IMPN  (0x5 << 14)
#define CTRL_ZQ_FORCE_IMPP  (0x6 << 17)
#define CTRL_DCC            (0xE38 << 20)
#define ZQ_CONTROL_VAL      (CTRL_ZQ_MODE_NOTERM | CTRL_ZQ_START\
        | CTRL_ZQ_DIV | CTRL_ZQ_MODE_DDS\
        | CTRL_ZQ_MODE_TERM | CTRL_ZQ_FORCE_IMPN\
        | CTRL_ZQ_FORCE_IMPP | CTRL_DCC)

#define ASYNC               (0 << 0)
#define CLK_RATIO           (1 << 1)
#define DIV_PIPE            (1 << 3)
#define AWR_ON              (1 << 4)
#define AREF_DISABLE        (0 << 5)
#define DRV_TYPE_DISABLE    (0 << 6)
#define CHIP0_NOT_EMPTY     (0 << 8)
#define CHIP1_NOT_EMPTY     (0 << 9)
#define DQ_SWAP_DISABLE     (0 << 10)
#define QOS_FAST_DISABLE    (0 << 11)
#define RD_FETCH            (0x3 << 12)
#define TIMEOUT_LEVEL0      (0xFFF << 16)
#define CONCONTROL_VAL      (ASYNC | CLK_RATIO | DIV_PIPE | AWR_ON\
        | AREF_DISABLE | DRV_TYPE_DISABLE\
        | CHIP0_NOT_EMPTY | CHIP1_NOT_EMPTY\
        | DQ_SWAP_DISABLE | QOS_FAST_DISABLE\
        | RD_FETCH | TIMEOUT_LEVEL0)

#define MEMCONTROL_OR       (1 | 2 | (1 << 4) | (1 << 24))
#define CLK_STOP_DISABLE    (0 << 1)
#define DPWRDN_DISABLE      (0 << 2)
#define DPWRDN_TYPE         (0 << 3)
#define TP_DISABLE          (0 << 4)
#define DSREF_DIABLE        (0 << 5)
#define ADD_LAT_PALL        (1 << 6)
#define MEM_TYPE_DDR3       (6 << 8)
#define MEM_WIDTH_32        (2 << 12)
#define NUM_CHIP_2          (1 << 16)
#define BL_8                (3 << 20)
#define MEMCONTROL_VAL      (CLK_STOP_DISABLE | DPWRDN_DISABLE\
        | DPWRDN_TYPE | TP_DISABLE | DSREF_DIABLE\
        | ADD_LAT_PALL | MEM_TYPE_DDR3 | MEM_WIDTH_32\
        | NUM_CHIP_2 | BL_8)


#define CHIP_BANK_8             (0x3 << 0)
#define CHIP_COL_10             (0x3 << 8)
#define CHIP_MAP_INTERLEAVED    (0x1 << 12)

#ifdef CONFIG_MIU_LINEAR
#define CHIP_ROW                (0x2 << 4)
#define CHIP0_BASE              (0x40 << 24)
#define CHIP1_BASE              (0x60 << 24)
#define CHIP_MASK               (0xe0 << 16)
#else
#define CHIP_ROW                (0x3 << 4)
#define CHIP0_BASE              (0x40 << 24)
#define CHIP1_BASE              (0x80 << 24)
#define CHIP_MASK               (0x80 << 16)
#endif

#define MEMCONFIG0_VAL      (CHIP_BANK_8 | CHIP_ROW | CHIP_COL_10\
        | CHIP_MAP_INTERLEAVED | CHIP_MASK | CHIP0_BASE)
#define MEMCONFIG1_VAL      (CHIP_BANK_8 | CHIP_ROW | CHIP_COL_10\
        | CHIP_MAP_INTERLEAVED | CHIP_MASK | CHIP1_BASE)

#define TP_CNT              (0x64 << 24)
#define PRECHCONFIG         TP_CNT

#define CTRL_OFF            (0 << 0)
#define CTRL_DLL_OFF        (0 << 1)
#define CTRL_HALF           (0 << 2)
#define CTRL_DFDQS          (1 << 3)
#define DQS_DELAY           (0 << 4)
#define CTRL_START_POINT    (0x10 << 8)
#define CTRL_INC            (0x10 << 16)
#define CTRL_FORCE          (0x71 << 24)
#define CONTROL0_VAL        (CTRL_OFF | CTRL_DLL_OFF | CTRL_HALF\
        | CTRL_DFDQS | DQS_DELAY | CTRL_START_POINT\
        | CTRL_INC | CTRL_FORCE)

#define CTRL_SHIFTC         (6 << 0)
#define CTRL_REF            (8 << 4)
#define CTRL_SHGATE         (1 << 29)
#define TERM_READ_EN        (1 << 30)
#define TERM_WRITE_EN       (1 << 31)
#define CONTROL1_VAL        (CTRL_SHIFTC | CTRL_REF | CTRL_SHGATE\
        | TERM_READ_EN | TERM_WRITE_EN)

#define CONTROL2_VAL        0x00000000

#ifdef DRAM_CLK_200
#define TIMINGREF_VAL       0x000000BB
#define TIMINGROW_VAL       0x4046654f
#define TIMINGDATA_VAL      0x46400506
#define TIMINGPOWER_VAL     0x52000A3C
#endif

#ifdef DRAM_CLK_330
#define TIMINGREF_VAL       0x000000BC
#define TIMINGROW_VAL       0x3545548d
#define TIMINGDATA_VAL      0x45430506
#define TIMINGPOWER_VAL     0x4439033c
#endif

#ifdef DRAM_CLK_400
#define TIMINGREF_VAL       0x000000BC
#define TIMINGROW_VAL       0x45430506
#define TIMINGDATA_VAL      0x56500506
#define TIMINGPOWER_VAL     0x5444033d
#endif

#endif

修改include/configs/itop4412.h,选择合适的时钟#define CONFIG_CLK_1000_200_200

四、验证结果

读写验证:在对应的内存区域随机选择内存地址,直接读写,看能否正常读写。

内存映射:在地址0x40000000处写入数据,在0x80000000读取数据,看其值是否一样。

内存间歇:在0x80000000附近读写内存,看内存是否连续。

内存长度:在0xC0000000附近读写内存,看内存实际容量。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容

  • feisky云计算、虚拟化与Linux技术笔记posts - 1014, comments - 298, trac...
    不排版阅读 3,837评论 0 5
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,938评论 6 13
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 这是一个充满选择的世界,人们每时每刻都在做着选择,小如每天的吃穿用度,大到读大学、找工作、寻找人生方向、确定人生目...
    抓瞎的猫阅读 222评论 0 0
  • 恩佐…恩佐… 之前看过《忠犬八公的故事》,后来又看了《拯救雪碧》。 最近看了《一条狗的使命》,现在看这本书。很神奇...
    小小棠梨阅读 1,048评论 0 0