/* * PD Buddy Firmware Library - USB Power Delivery for everyone * Copyright 2017-2018 Clayton G. Hobbs * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "int_n.h" #include "BSP.h" #include "BSP_PD.h" #include "fusb302b.h" #include "fusbpd.h" #include "policy_engine.h" #include "protocol_tx.h" #include "task.h" #include #include volatile osThreadId InterruptHandler::TaskHandle = NULL; uint32_t InterruptHandler::TaskBuffer[InterruptHandler::TaskStackSize]; osStaticThreadDef_t InterruptHandler::TaskControlBlock; union pd_msg InterruptHandler::tempMessage; void InterruptHandler::init() { TaskHandle = NULL; osThreadStaticDef(intTask, Thread, PDB_PRIO_PRL_INT_N, 0, TaskStackSize, TaskBuffer, &TaskControlBlock); TaskHandle = osThreadCreate(osThread(intTask), NULL); } void InterruptHandler::readPendingMessage() { /* Get a buffer to read the message into. Guaranteed to not fail * because we have a big enough pool and are careful. */ memset(&tempMessage, 0, sizeof(tempMessage)); /* Read the message */ fusb_read_message(&tempMessage); /* If it's a Soft_Reset, go to the soft reset state */ if (PD_MSGTYPE_GET(&tempMessage) == PD_MSGTYPE_SOFT_RESET && PD_NUMOBJ_GET(&tempMessage) == 0) { /* TX transitions to its reset state */ ProtocolTransmit::notify(ProtocolTransmit::Notifications::PDB_EVT_PRLTX_RESET); } else { /* Tell ProtocolTX to discard the message being transmitted */ ProtocolTransmit::notify(ProtocolTransmit::Notifications::PDB_EVT_PRLTX_DISCARD); /* Pass the message to the policy engine. */ PolicyEngine::handleMessage(&tempMessage); } } void InterruptHandler::Thread(const void *arg) { (void)arg; union fusb_status status; for (;;) { // If the irq is low continue, otherwise wait for irq or timeout if (!getFUS302IRQLow()) { xTaskNotifyWait(0x00, 0x0F, NULL, TICKS_SECOND * 30); } /* Read the FUSB302B status and interrupt registers */ fusb_get_status(&status); /* If the I_GCRCSENT flag is set, tell the Protocol RX thread */ // This means a message was recieved with a good CRC if (status.interruptb & FUSB_INTERRUPTB_I_GCRCSENT) { readPendingMessage(); } /* If the I_TXSENT or I_RETRYFAIL flag is set, tell the Protocol TX * thread */ if (status.interrupta & FUSB_INTERRUPTA_I_TXSENT) { ProtocolTransmit::notify(ProtocolTransmit::Notifications::PDB_EVT_PRLTX_I_TXSENT); } if (status.interrupta & FUSB_INTERRUPTA_I_RETRYFAIL) { ProtocolTransmit::notify(ProtocolTransmit::Notifications::PDB_EVT_PRLTX_I_RETRYFAIL); } /* If the I_OCP_TEMP and OVRTEMP flags are set, tell the Policy * Engine thread */ if ((status.interrupta & FUSB_INTERRUPTA_I_OCP_TEMP) && (status.status1 & FUSB_STATUS1_OVRTEMP)) { PolicyEngine::notify(PolicyEngine::Notifications::PDB_EVT_PE_I_OVRTEMP); } } } void InterruptHandler::irqCallback() { if (TaskHandle != NULL) { BaseType_t taskWoke = pdFALSE; xTaskNotifyFromISR(TaskHandle, 0x01, eNotifyAction::eSetBits, &taskWoke); portYIELD_FROM_ISR(taskWoke); } }