static void adp_sense_timeout(void *ptr) { dwc_otg_core_if_t *core_if = (dwc_otg_core_if_t *) ptr; core_if->adp.sense_timer_started = 0; DWC_PRINTF("ADP SENSE TIMEOUT\n"); if (core_if->adp_enable) { dwc_otg_adp_sense_stop(core_if); dwc_otg_adp_probe_start(core_if); } }
/** * This function is called when the ADP vbus timer expires. Timeout is 1.1s. */ static void adp_vbuson_timeout(void *ptr) { gpwrdn_data_t gpwrdn; dwc_otg_core_if_t *core_if = (dwc_otg_core_if_t *) ptr; hprt0_data_t hprt0 = {.d32 = 0 }; pcgcctl_data_t pcgcctl = {.d32 = 0 }; DWC_PRINTF("%s: 1.1 seconds expire after turning on VBUS\n",__FUNCTION__); if (core_if) { core_if->adp.vbuson_timer_started = 0; /* Turn off vbus */ hprt0.b.prtpwr = 1; DWC_MODIFY_REG32(core_if->host_if->hprt0, hprt0.d32, 0); gpwrdn.d32 = 0; /* Power off the core */ if (core_if->power_down == 2) { /* Enable Wakeup Logic */ /* gpwrdn.b.wkupactiv = 1; */ gpwrdn.b.pmuactv = 0; gpwrdn.b.pwrdnrstn = 1; gpwrdn.b.pwrdnclmp = 1; DWC_MODIFY_REG32(&core_if->core_global_regs->gpwrdn, 0, gpwrdn.d32); /* Suspend the Phy Clock */ pcgcctl.b.stoppclk = 1; DWC_MODIFY_REG32(core_if->pcgcctl, 0, pcgcctl.d32); /* Switch on VDD */ /* gpwrdn.b.wkupactiv = 1;*/ gpwrdn.b.pmuactv = 1; gpwrdn.b.pwrdnrstn = 1; gpwrdn.b.pwrdnclmp = 1; DWC_MODIFY_REG32(&core_if->core_global_regs->gpwrdn, 0, gpwrdn.d32); } else { /* Enable Power Down Logic */ gpwrdn.b.pmuintsel = 1; gpwrdn.b.pmuactv = 1; DWC_MODIFY_REG32(&core_if->core_global_regs->gpwrdn, 0, gpwrdn.d32); } /* Power off the core */ if (core_if->power_down == 2) { gpwrdn.d32 = 0; gpwrdn.b.pwrdnswtch = 1; DWC_MODIFY_REG32(&core_if->core_global_regs->gpwrdn, gpwrdn.d32, 0); } /* Unmask SRP detected interrupt from Power Down Logic */ gpwrdn.d32 = 0; gpwrdn.b.srp_det_msk = 1; DWC_MODIFY_REG32(&core_if->core_global_regs->gpwrdn, 0, gpwrdn.d32); dwc_otg_adp_probe_start(core_if); dwc_otg_dump_global_registers(core_if); dwc_otg_dump_host_registers(core_if); } } /** * Start the ADP Initial Probe timer to detect if Port Connected interrupt is * not asserted within 1.1 seconds. * * @param core_if the pointer to core_if strucure. */ void dwc_otg_adp_vbuson_timer_start(dwc_otg_core_if_t * core_if) { core_if->adp.vbuson_timer_started = 1; if (core_if->adp.vbuson_timer) { DWC_PRINTF("SCHEDULING VBUSON TIMER\n"); /* 1.1 secs + 60ms necessary for cil_hcd_start*/ DWC_TIMER_SCHEDULE(core_if->adp.vbuson_timer, 1160); } else { DWC_WARN("VBUSON_TIMER = %p\n",core_if->adp.vbuson_timer); } } #if 0 /** * Masks all DWC OTG core interrupts * */ static void mask_all_interrupts(dwc_otg_core_if_t * core_if) { int i; gahbcfg_data_t ahbcfg = {.d32 = 0 }; /* Mask Host Interrupts */ /* Clear and disable HCINTs */ for (i = 0; i < core_if->core_params->host_channels; i++) { DWC_WRITE_REG32(&core_if->host_if->hc_regs[i]->hcintmsk, 0); DWC_WRITE_REG32(&core_if->host_if->hc_regs[i]->hcint, 0xFFFFFFFF); } /* Clear and disable HAINT */ DWC_WRITE_REG32(&core_if->host_if->host_global_regs->haintmsk, 0x0000); DWC_WRITE_REG32(&core_if->host_if->host_global_regs->haint, 0xFFFFFFFF); /* Mask Device Interrupts */ if (!core_if->multiproc_int_enable) { /* Clear and disable IN Endpoint interrupts */ DWC_WRITE_REG32(&core_if->dev_if->dev_global_regs->diepmsk, 0); for (i = 0; i <= core_if->dev_if->num_in_eps; i++) { DWC_WRITE_REG32(&core_if->dev_if->in_ep_regs[i]-> diepint, 0xFFFFFFFF); } /* Clear and disable OUT Endpoint interrupts */ DWC_WRITE_REG32(&core_if->dev_if->dev_global_regs->doepmsk, 0); for (i = 0; i <= core_if->dev_if->num_out_eps; i++) { DWC_WRITE_REG32(&core_if->dev_if->out_ep_regs[i]-> doepint, 0xFFFFFFFF); } /* Clear and disable DAINT */ DWC_WRITE_REG32(&core_if->dev_if->dev_global_regs->daint, 0xFFFFFFFF); DWC_WRITE_REG32(&core_if->dev_if->dev_global_regs->daintmsk, 0); } else { for (i = 0; i < core_if->dev_if->num_in_eps; ++i) { DWC_WRITE_REG32(&core_if->dev_if->dev_global_regs-> diepeachintmsk[i], 0); DWC_WRITE_REG32(&core_if->dev_if->in_ep_regs[i]-> diepint, 0xFFFFFFFF); } for (i = 0; i < core_if->dev_if->num_out_eps; ++i) { DWC_WRITE_REG32(&core_if->dev_if->dev_global_regs-> doepeachintmsk[i], 0); DWC_WRITE_REG32(&core_if->dev_if->out_ep_regs[i]-> doepint, 0xFFFFFFFF); } DWC_WRITE_REG32(&core_if->dev_if->dev_global_regs->deachintmsk, 0); DWC_WRITE_REG32(&core_if->dev_if->dev_global_regs->deachint, 0xFFFFFFFF); } /* Disable interrupts */ ahbcfg.b.glblintrmsk = 1; DWC_MODIFY_REG32(&core_if->core_global_regs->gahbcfg, ahbcfg.d32, 0); /* Disable all interrupts. */ DWC_WRITE_REG32(&core_if->core_global_regs->gintmsk, 0); /* Clear any pending interrupts */ DWC_WRITE_REG32(&core_if->core_global_regs->gintsts, 0xFFFFFFFF); /* Clear any pending OTG Interrupts */ DWC_WRITE_REG32(&core_if->core_global_regs->gotgint, 0xFFFFFFFF); } /** * Unmask Port Connection Detected interrupt * */ static void unmask_conn_det_intr(dwc_otg_core_if_t * core_if) { gintmsk_data_t gintmsk = {.d32 = 0,.b.portintr = 1 }; DWC_WRITE_REG32(&core_if->core_global_regs->gintmsk, gintmsk.d32); } #endif /** * Starts the ADP Probing * * @param core_if the pointer to core_if structure. */ uint32_t dwc_otg_adp_probe_start(dwc_otg_core_if_t * core_if) { adpctl_data_t adpctl = {.d32 = 0}; gpwrdn_data_t gpwrdn; #if 0 adpctl_data_t adpctl_int = {.d32 = 0, .b.adp_prb_int = 1, .b.adp_sns_int = 1, b.adp_tmout_int}; #endif dwc_otg_disable_global_interrupts(core_if); DWC_PRINTF("ADP Probe Start\n"); core_if->adp.probe_enabled = 1; adpctl.b.adpres = 1; dwc_otg_adp_write_reg(core_if, adpctl.d32); while (adpctl.b.adpres) { adpctl.d32 = dwc_otg_adp_read_reg(core_if); } adpctl.d32 = 0; gpwrdn.d32 = DWC_READ_REG32(&core_if->core_global_regs->gpwrdn); /* In Host mode unmask SRP detected interrupt */ gpwrdn.d32 = 0; gpwrdn.b.sts_chngint_msk = 1; if (!gpwrdn.b.idsts) { gpwrdn.b.srp_det_msk = 1; } DWC_MODIFY_REG32(&core_if->core_global_regs->gpwrdn, 0, gpwrdn.d32); adpctl.b.adp_tmout_int_msk = 1; adpctl.b.adp_prb_int_msk = 1; adpctl.b.prb_dschg = 1; adpctl.b.prb_delta = 1; adpctl.b.prb_per = 1; adpctl.b.adpen = 1; adpctl.b.enaprb = 1; dwc_otg_adp_write_reg(core_if, adpctl.d32); DWC_PRINTF("ADP Probe Finish\n"); return 0; } /** * Starts the ADP Sense timer to detect if ADP Sense interrupt is not asserted * within 3 seconds. * * @param core_if the pointer to core_if strucure. */ void dwc_otg_adp_sense_timer_start(dwc_otg_core_if_t * core_if) { core_if->adp.sense_timer_started = 1; DWC_TIMER_SCHEDULE(core_if->adp.sense_timer, 3000 /* 3 secs */ ); } /** * Starts the ADP Sense * * @param core_if the pointer to core_if strucure. */ uint32_t dwc_otg_adp_sense_start(dwc_otg_core_if_t * core_if) { adpctl_data_t adpctl; DWC_PRINTF("ADP Sense Start\n"); /* Unmask ADP sense interrupt and mask all other from the core */ adpctl.d32 = dwc_otg_adp_read_reg_filter(core_if); adpctl.b.adp_sns_int_msk = 1; dwc_otg_adp_write_reg(core_if, adpctl.d32); dwc_otg_disable_global_interrupts(core_if); // vahrama /* Set ADP reset bit*/ adpctl.d32 = dwc_otg_adp_read_reg_filter(core_if); adpctl.b.adpres = 1; dwc_otg_adp_write_reg(core_if, adpctl.d32); while (adpctl.b.adpres) { adpctl.d32 = dwc_otg_adp_read_reg(core_if); } adpctl.b.adpres = 0; adpctl.b.adpen = 1; adpctl.b.enasns = 1; dwc_otg_adp_write_reg(core_if, adpctl.d32); dwc_otg_adp_sense_timer_start(core_if); return 0; }
static void adp_probe_func(void *ptr) { dwc_otg_core_if_t *core_if = (dwc_otg_core_if_t *) ptr; dwc_otg_adp_probe_start(core_if); }