|  | /* | 
|  | * Low level PM code for TI EMIF | 
|  | * | 
|  | * Copyright (C) 2016-2017 Texas Instruments Incorporated - http://www.ti.com/ | 
|  | *	Dave Gerlach | 
|  | * | 
|  | * 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 version 2. | 
|  | * | 
|  | * This program is distributed "as is" WITHOUT ANY WARRANTY of any | 
|  | * kind, whether express or implied; without even the implied warranty | 
|  | * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | * GNU General Public License for more details. | 
|  | */ | 
|  |  | 
|  | #include <generated/ti-emif-asm-offsets.h> | 
|  | #include <linux/linkage.h> | 
|  | #include <asm/assembler.h> | 
|  | #include <asm/memory.h> | 
|  |  | 
|  | #include "emif.h" | 
|  |  | 
|  | #define EMIF_POWER_MGMT_WAIT_SELF_REFRESH_8192_CYCLES	0x00a0 | 
|  | #define EMIF_POWER_MGMT_SR_TIMER_MASK			0x00f0 | 
|  | #define EMIF_POWER_MGMT_SELF_REFRESH_MODE		0x0200 | 
|  | #define EMIF_POWER_MGMT_SELF_REFRESH_MODE_MASK		0x0700 | 
|  |  | 
|  | #define EMIF_SDCFG_TYPE_DDR2				0x2 << SDRAM_TYPE_SHIFT | 
|  | #define EMIF_STATUS_READY				0x4 | 
|  |  | 
|  | #define AM43XX_EMIF_PHY_CTRL_REG_COUNT                  0x120 | 
|  |  | 
|  | #define EMIF_AM437X_REGISTERS				0x1 | 
|  |  | 
|  | .arm | 
|  | .align 3 | 
|  |  | 
|  | ENTRY(ti_emif_sram) | 
|  |  | 
|  | /* | 
|  | * void ti_emif_save_context(void) | 
|  | * | 
|  | * Used during suspend to save the context of all required EMIF registers | 
|  | * to local memory if the EMIF is going to lose context during the sleep | 
|  | * transition. Operates on the VIRTUAL address of the EMIF. | 
|  | */ | 
|  | ENTRY(ti_emif_save_context) | 
|  | stmfd   sp!, {r4 - r11, lr}     @ save registers on stack | 
|  |  | 
|  | adr	r4, ti_emif_pm_sram_data | 
|  | ldr	r0, [r4, #EMIF_PM_BASE_ADDR_VIRT_OFFSET] | 
|  | ldr	r2, [r4, #EMIF_PM_REGS_VIRT_OFFSET] | 
|  |  | 
|  | /* Save EMIF configuration */ | 
|  | ldr	r1, [r0, #EMIF_SDRAM_CONFIG] | 
|  | str	r1, [r2, #EMIF_SDCFG_VAL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_SDRAM_REFRESH_CONTROL] | 
|  | str	r1, [r2, #EMIF_REF_CTRL_VAL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_SDRAM_TIMING_1] | 
|  | str     r1, [r2, #EMIF_TIMING1_VAL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_SDRAM_TIMING_2] | 
|  | str     r1, [r2, #EMIF_TIMING2_VAL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_SDRAM_TIMING_3] | 
|  | str     r1, [r2, #EMIF_TIMING3_VAL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_POWER_MANAGEMENT_CONTROL] | 
|  | str     r1, [r2, #EMIF_PMCR_VAL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_POWER_MANAGEMENT_CTRL_SHDW] | 
|  | str     r1, [r2, #EMIF_PMCR_SHDW_VAL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_SDRAM_OUTPUT_IMPEDANCE_CALIBRATION_CONFIG] | 
|  | str     r1, [r2, #EMIF_ZQCFG_VAL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_DDR_PHY_CTRL_1] | 
|  | str     r1, [r2, #EMIF_DDR_PHY_CTLR_1_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_COS_CONFIG] | 
|  | str     r1, [r2, #EMIF_COS_CONFIG_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_PRIORITY_TO_CLASS_OF_SERVICE_MAPPING] | 
|  | str     r1, [r2, #EMIF_PRIORITY_TO_COS_MAPPING_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_CONNECTION_ID_TO_CLASS_OF_SERVICE_1_MAPPING] | 
|  | str     r1, [r2, #EMIF_CONNECT_ID_SERV_1_MAP_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_CONNECTION_ID_TO_CLASS_OF_SERVICE_2_MAPPING] | 
|  | str     r1, [r2, #EMIF_CONNECT_ID_SERV_2_MAP_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_OCP_CONFIG] | 
|  | str     r1, [r2, #EMIF_OCP_CONFIG_VAL_OFFSET] | 
|  |  | 
|  | ldr	r5, [r4, #EMIF_PM_CONFIG_OFFSET] | 
|  | cmp	r5, #EMIF_SRAM_AM43_REG_LAYOUT | 
|  | bne	emif_skip_save_extra_regs | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_READ_WRITE_LEVELING_RAMP_CONTROL] | 
|  | str     r1, [r2, #EMIF_RD_WR_LEVEL_RAMP_CTRL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_READ_WRITE_EXECUTION_THRESHOLD] | 
|  | str     r1, [r2, #EMIF_RD_WR_EXEC_THRESH_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_LPDDR2_NVM_TIMING] | 
|  | str     r1, [r2, #EMIF_LPDDR2_NVM_TIM_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_LPDDR2_NVM_TIMING_SHDW] | 
|  | str     r1, [r2, #EMIF_LPDDR2_NVM_TIM_SHDW_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_DLL_CALIB_CTRL] | 
|  | str     r1, [r2, #EMIF_DLL_CALIB_CTRL_VAL_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_DLL_CALIB_CTRL_SHDW] | 
|  | str     r1, [r2, #EMIF_DLL_CALIB_CTRL_VAL_SHDW_OFFSET] | 
|  |  | 
|  | /* Loop and save entire block of emif phy regs */ | 
|  | mov	r5, #0x0 | 
|  | add	r4, r2, #EMIF_EXT_PHY_CTRL_VALS_OFFSET | 
|  | add	r3, r0, #EMIF_EXT_PHY_CTRL_1 | 
|  | ddr_phy_ctrl_save: | 
|  | ldr	r1, [r3, r5] | 
|  | str	r1, [r4, r5] | 
|  | add	r5, r5, #0x4 | 
|  | cmp	r5, #AM43XX_EMIF_PHY_CTRL_REG_COUNT | 
|  | bne	ddr_phy_ctrl_save | 
|  |  | 
|  | emif_skip_save_extra_regs: | 
|  | ldmfd	sp!, {r4 - r11, pc}	@ restore regs and return | 
|  | ENDPROC(ti_emif_save_context) | 
|  |  | 
|  | /* | 
|  | * void ti_emif_restore_context(void) | 
|  | * | 
|  | * Used during resume to restore the context of all required EMIF registers | 
|  | * from local memory after the EMIF has lost context during a sleep transition. | 
|  | * Operates on the PHYSICAL address of the EMIF. | 
|  | */ | 
|  | ENTRY(ti_emif_restore_context) | 
|  | adr	r4, ti_emif_pm_sram_data | 
|  | ldr	r0, [r4, #EMIF_PM_BASE_ADDR_PHYS_OFFSET] | 
|  | ldr	r2, [r4, #EMIF_PM_REGS_PHYS_OFFSET] | 
|  |  | 
|  | /* Config EMIF Timings */ | 
|  | ldr     r1, [r2, #EMIF_DDR_PHY_CTLR_1_OFFSET] | 
|  | str	r1, [r0, #EMIF_DDR_PHY_CTRL_1] | 
|  | str	r1, [r0, #EMIF_DDR_PHY_CTRL_1_SHDW] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_TIMING1_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_SDRAM_TIMING_1] | 
|  | str	r1, [r0, #EMIF_SDRAM_TIMING_1_SHDW] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_TIMING2_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_SDRAM_TIMING_2] | 
|  | str	r1, [r0, #EMIF_SDRAM_TIMING_2_SHDW] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_TIMING3_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_SDRAM_TIMING_3] | 
|  | str	r1, [r0, #EMIF_SDRAM_TIMING_3_SHDW] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_REF_CTRL_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_SDRAM_REFRESH_CONTROL] | 
|  | str	r1, [r0, #EMIF_SDRAM_REFRESH_CTRL_SHDW] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_PMCR_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_POWER_MANAGEMENT_CONTROL] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_PMCR_SHDW_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_POWER_MANAGEMENT_CTRL_SHDW] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_COS_CONFIG_OFFSET] | 
|  | str	r1, [r0, #EMIF_COS_CONFIG] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_PRIORITY_TO_COS_MAPPING_OFFSET] | 
|  | str	r1, [r0, #EMIF_PRIORITY_TO_CLASS_OF_SERVICE_MAPPING] | 
|  |  | 
|  | ldr	r1, [r2, #EMIF_CONNECT_ID_SERV_1_MAP_OFFSET] | 
|  | str	r1, [r0, #EMIF_CONNECTION_ID_TO_CLASS_OF_SERVICE_1_MAPPING] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_CONNECT_ID_SERV_2_MAP_OFFSET] | 
|  | str	r1, [r0, #EMIF_CONNECTION_ID_TO_CLASS_OF_SERVICE_2_MAPPING] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_OCP_CONFIG_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_OCP_CONFIG] | 
|  |  | 
|  | ldr	r5, [r4, #EMIF_PM_CONFIG_OFFSET] | 
|  | cmp	r5, #EMIF_SRAM_AM43_REG_LAYOUT | 
|  | bne	emif_skip_restore_extra_regs | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_RD_WR_LEVEL_RAMP_CTRL_OFFSET] | 
|  | str	r1, [r0, #EMIF_READ_WRITE_LEVELING_RAMP_CONTROL] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_RD_WR_EXEC_THRESH_OFFSET] | 
|  | str	r1, [r0, #EMIF_READ_WRITE_EXECUTION_THRESHOLD] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_LPDDR2_NVM_TIM_OFFSET] | 
|  | str	r1, [r0, #EMIF_LPDDR2_NVM_TIMING] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_LPDDR2_NVM_TIM_SHDW_OFFSET] | 
|  | str	r1, [r0, #EMIF_LPDDR2_NVM_TIMING_SHDW] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_DLL_CALIB_CTRL_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_DLL_CALIB_CTRL] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_DLL_CALIB_CTRL_VAL_SHDW_OFFSET] | 
|  | str	r1, [r0, #EMIF_DLL_CALIB_CTRL_SHDW] | 
|  |  | 
|  | ldr     r1, [r2, #EMIF_ZQCFG_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_SDRAM_OUTPUT_IMPEDANCE_CALIBRATION_CONFIG] | 
|  |  | 
|  | /* Loop and restore entire block of emif phy regs */ | 
|  | mov	r5, #0x0 | 
|  | /* Load ti_emif_regs_amx3 + EMIF_EXT_PHY_CTRL_VALS_OFFSET for address | 
|  | * to phy register save space | 
|  | */ | 
|  | add	r3, r2, #EMIF_EXT_PHY_CTRL_VALS_OFFSET | 
|  | add	r4, r0, #EMIF_EXT_PHY_CTRL_1 | 
|  | ddr_phy_ctrl_restore: | 
|  | ldr	r1, [r3, r5] | 
|  | str	r1, [r4, r5] | 
|  | add	r5, r5, #0x4 | 
|  | cmp	r5, #AM43XX_EMIF_PHY_CTRL_REG_COUNT | 
|  | bne	ddr_phy_ctrl_restore | 
|  |  | 
|  | emif_skip_restore_extra_regs: | 
|  | /* | 
|  | * Output impedence calib needed only for DDR3 | 
|  | * but since the initial state of this will be | 
|  | * disabled for DDR2 no harm in restoring the | 
|  | * old configuration | 
|  | */ | 
|  | ldr     r1, [r2, #EMIF_ZQCFG_VAL_OFFSET] | 
|  | str	r1, [r0, #EMIF_SDRAM_OUTPUT_IMPEDANCE_CALIBRATION_CONFIG] | 
|  |  | 
|  | /* Write to sdcfg last for DDR2 only */ | 
|  | ldr	r1, [r2, #EMIF_SDCFG_VAL_OFFSET] | 
|  | and	r2, r1, #SDRAM_TYPE_MASK | 
|  | cmp	r2, #EMIF_SDCFG_TYPE_DDR2 | 
|  | streq	r1, [r0, #EMIF_SDRAM_CONFIG] | 
|  |  | 
|  | mov	pc, lr | 
|  | ENDPROC(ti_emif_restore_context) | 
|  |  | 
|  | /* | 
|  | * void ti_emif_enter_sr(void) | 
|  | * | 
|  | * Programs the EMIF to tell the SDRAM to enter into self-refresh | 
|  | * mode during a sleep transition. Operates on the VIRTUAL address | 
|  | * of the EMIF. | 
|  | */ | 
|  | ENTRY(ti_emif_enter_sr) | 
|  | stmfd   sp!, {r4 - r11, lr}     @ save registers on stack | 
|  |  | 
|  | adr	r4, ti_emif_pm_sram_data | 
|  | ldr	r0, [r4, #EMIF_PM_BASE_ADDR_VIRT_OFFSET] | 
|  | ldr	r2, [r4, #EMIF_PM_REGS_VIRT_OFFSET] | 
|  |  | 
|  | ldr	r1, [r0, #EMIF_POWER_MANAGEMENT_CONTROL] | 
|  | bic	r1, r1, #EMIF_POWER_MGMT_SELF_REFRESH_MODE_MASK | 
|  | orr	r1, r1, #EMIF_POWER_MGMT_SELF_REFRESH_MODE | 
|  | str	r1, [r0, #EMIF_POWER_MANAGEMENT_CONTROL] | 
|  |  | 
|  | ldmfd	sp!, {r4 - r11, pc}	@ restore regs and return | 
|  | ENDPROC(ti_emif_enter_sr) | 
|  |  | 
|  | /* | 
|  | * void ti_emif_exit_sr(void) | 
|  | * | 
|  | * Programs the EMIF to tell the SDRAM to exit self-refresh mode | 
|  | * after a sleep transition. Operates on the PHYSICAL address of | 
|  | * the EMIF. | 
|  | */ | 
|  | ENTRY(ti_emif_exit_sr) | 
|  | adr	r4, ti_emif_pm_sram_data | 
|  | ldr	r0, [r4, #EMIF_PM_BASE_ADDR_PHYS_OFFSET] | 
|  | ldr	r2, [r4, #EMIF_PM_REGS_PHYS_OFFSET] | 
|  |  | 
|  | /* | 
|  | * Toggle EMIF to exit refresh mode: | 
|  | * if EMIF lost context, PWR_MGT_CTRL is currently 0, writing disable | 
|  | *   (0x0), wont do diddly squat! so do a toggle from SR(0x2) to disable | 
|  | *   (0x0) here. | 
|  | * *If* EMIF did not lose context, nothing broken as we write the same | 
|  | *   value(0x2) to reg before we write a disable (0x0). | 
|  | */ | 
|  | ldr	r1, [r2, #EMIF_PMCR_VAL_OFFSET] | 
|  | bic	r1, r1, #EMIF_POWER_MGMT_SELF_REFRESH_MODE_MASK | 
|  | orr	r1, r1, #EMIF_POWER_MGMT_SELF_REFRESH_MODE | 
|  | str	r1, [r0, #EMIF_POWER_MANAGEMENT_CONTROL] | 
|  | bic	r1, r1, #EMIF_POWER_MGMT_SELF_REFRESH_MODE_MASK | 
|  | str	r1, [r0, #EMIF_POWER_MANAGEMENT_CONTROL] | 
|  |  | 
|  | /* Wait for EMIF to become ready */ | 
|  | 1:	ldr     r1, [r0, #EMIF_STATUS] | 
|  | tst     r1, #EMIF_STATUS_READY | 
|  | beq     1b | 
|  |  | 
|  | mov	pc, lr | 
|  | ENDPROC(ti_emif_exit_sr) | 
|  |  | 
|  | /* | 
|  | * void ti_emif_abort_sr(void) | 
|  | * | 
|  | * Disables self-refresh after a failed transition to a low-power | 
|  | * state so the kernel can jump back to DDR and follow abort path. | 
|  | * Operates on the VIRTUAL address of the EMIF. | 
|  | */ | 
|  | ENTRY(ti_emif_abort_sr) | 
|  | stmfd   sp!, {r4 - r11, lr}     @ save registers on stack | 
|  |  | 
|  | adr	r4, ti_emif_pm_sram_data | 
|  | ldr	r0, [r4, #EMIF_PM_BASE_ADDR_VIRT_OFFSET] | 
|  | ldr	r2, [r4, #EMIF_PM_REGS_VIRT_OFFSET] | 
|  |  | 
|  | ldr	r1, [r2, #EMIF_PMCR_VAL_OFFSET] | 
|  | bic	r1, r1, #EMIF_POWER_MGMT_SELF_REFRESH_MODE_MASK | 
|  | str	r1, [r0, #EMIF_POWER_MANAGEMENT_CONTROL] | 
|  |  | 
|  | /* Wait for EMIF to become ready */ | 
|  | 1:	ldr     r1, [r0, #EMIF_STATUS] | 
|  | tst     r1, #EMIF_STATUS_READY | 
|  | beq     1b | 
|  |  | 
|  | ldmfd	sp!, {r4 - r11, pc}	@ restore regs and return | 
|  | ENDPROC(ti_emif_abort_sr) | 
|  |  | 
|  | .align 3 | 
|  | ENTRY(ti_emif_pm_sram_data) | 
|  | .space EMIF_PM_DATA_SIZE | 
|  | ENTRY(ti_emif_sram_sz) | 
|  | .word   . - ti_emif_save_context |