diff --git a/CMakeLists.txt b/CMakeLists.txt index 5aef66172..6219df302 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -160,7 +160,6 @@ set(header_eez_apps_psu src/eez/apps/psu/channel_dispatcher.h src/eez/apps/psu/conf.h src/eez/apps/psu/conf_advanced.h - src/eez/apps/psu/conf_channel.h src/eez/apps/psu/conf_user.h src/eez/apps/psu/datetime.h src/eez/apps/psu/debug.h diff --git a/images/Module DCP405b.png b/images/Module DCP405b.png index 7a74e0599..0dd4ad14b 100644 Binary files a/images/Module DCP405b.png and b/images/Module DCP405b.png differ diff --git a/modular-psu-firmware.eez-project b/modular-psu-firmware.eez-project index 81d252a93..6f817a017 100644 --- a/modular-psu-firmware.eez-project +++ b/modular-psu-firmware.eez-project @@ -1464,15 +1464,6 @@ "simulator" ] }, - { - "name": "slot_module_type", - "type": "enum", - "enumItems": "[\"None\", \"DCP505\", \"DCP405\"]", - "defaultValue": "1", - "usedIn": [ - "simulator" - ] - }, { "name": "simulator.loadState", "description": "", @@ -3140,6 +3131,7 @@ "type": "MultilineText", "style": { "inheritFrom": "default_S", + "alignHorizontal": "left", "color": "status_warning", "padding": 0 }, @@ -3157,6 +3149,7 @@ "type": "MultilineText", "style": { "inheritFrom": "default_S", + "alignHorizontal": "left", "padding": 0 }, "activeStyle": { @@ -8727,21 +8720,18 @@ "height": 28, "widgets": [ { - "type": "Text", + "type": "Rectangle", "style": { - "inheritFrom": "default_S", - "padding": 0 + "inheritFrom": "default" }, "activeStyle": { - "inheritFrom": "default", - "padding": 0 + "inheritFrom": "default" }, - "action": "", "left": 0, "top": 0, "width": 234, "height": 28, - "text": "Advanced options" + "invertColors": true }, { "type": "Text", @@ -15179,7 +15169,7 @@ "top": 0, "width": 166, "height": 32, - "text": "Display value #1:" + "text": "YT view value #1:" }, { "type": "DisplayData", @@ -15217,7 +15207,7 @@ "top": 36, "width": 166, "height": 32, - "text": "Display value #2:" + "text": "YT view Display value #2:" }, { "type": "DisplayData", @@ -15401,7 +15391,7 @@ "top": 0, "width": 321, "height": 32, - "text": "Set over-power protection" + "text": "Display view settings" } ] }, @@ -25866,9 +25856,8 @@ { "type": "Text", "style": { - "inheritFrom": "option_toggle_L_center", + "inheritFrom": "default", "font": "Oswald17", - "alignHorizontal": "center", "backgroundColor": "Background", "padding": "0 0 0 6" }, @@ -25885,7 +25874,7 @@ { "type": "Text", "style": { - "inheritFrom": "option_toggle_L_center", + "inheritFrom": "default", "font": "Oswald17", "alignHorizontal": "left", "padding": "0 0 0 " @@ -25936,7 +25925,6 @@ "style": { "inheritFrom": "option_toggle_L_center", "font": "Oswald17", - "alignHorizontal": "center", "padding": "0 0 0 6" }, "activeStyle": { @@ -25954,7 +25942,6 @@ "style": { "inheritFrom": "option_toggle_L_center", "font": "Oswald17", - "alignHorizontal": "left", "padding": "0 0 0 6" }, "activeStyle": { @@ -25991,39 +25978,41 @@ ] }, { - "type": "Container", + "type": "Select", "style": { "inheritFrom": "default" }, "activeStyle": { "inheritFrom": "default" }, - "action": "set_coupling_parallel", + "data": "is_coupling_parallel_allowed", "left": 5, "top": 120, "width": 189, "height": 36, "widgets": [ { - "type": "Select", + "type": "Container", "style": { "inheritFrom": "default" }, "activeStyle": { "inheritFrom": "default" }, - "data": "is_coupling_type_parallel", + "data": "", + "action": "", "left": 0, "top": 0, - "width": 38, + "width": 189, "height": 36, "widgets": [ { "type": "Text", "style": { - "inheritFrom": "option_toggle_L_center", + "inheritFrom": "default", "font": "Oswald17", "alignHorizontal": "center", + "backgroundColor": "Background", "padding": "0 0 0 6" }, "activeStyle": { @@ -26039,40 +26028,107 @@ { "type": "Text", "style": { - "inheritFrom": "option_toggle_L_center", + "inheritFrom": "default", "font": "Oswald17", "alignHorizontal": "left", - "padding": "0 0 0 6" + "padding": "0 0 0 " }, "activeStyle": { "inheritFrom": "default" }, "action": "", - "left": 0, + "left": 38, "top": 0, - "width": 38, + "width": 150, "height": 36, - "text": "\\u008e" + "text": "Parallel" } ] }, { - "type": "Text", + "type": "Container", "style": { - "inheritFrom": "option_toggle_L_center", - "font": "Oswald17", - "alignHorizontal": "left", - "padding": "0 0 0 " + "inheritFrom": "default" }, "activeStyle": { "inheritFrom": "default" }, - "action": "", - "left": 38, + "action": "set_coupling_parallel", + "left": 0, "top": 0, - "width": 150, + "width": 189, "height": 36, - "text": "Parallel" + "widgets": [ + { + "type": "Select", + "style": { + "inheritFrom": "default" + }, + "activeStyle": { + "inheritFrom": "default" + }, + "data": "is_coupling_type_parallel", + "left": 0, + "top": 0, + "width": 38, + "height": 36, + "widgets": [ + { + "type": "Text", + "style": { + "inheritFrom": "option_toggle_L_center", + "font": "Oswald17", + "alignHorizontal": "center", + "padding": "0 0 0 6" + }, + "activeStyle": { + "inheritFrom": "default" + }, + "action": "", + "left": 0, + "top": 0, + "width": 38, + "height": 36, + "text": "\\u008d" + }, + { + "type": "Text", + "style": { + "inheritFrom": "option_toggle_L_center", + "font": "Oswald17", + "padding": "0 0 0 6" + }, + "activeStyle": { + "inheritFrom": "default" + }, + "action": "", + "left": 0, + "top": 0, + "width": 38, + "height": 36, + "text": "\\u008e" + } + ] + }, + { + "type": "Text", + "style": { + "inheritFrom": "option_toggle_L_center", + "font": "Oswald17", + "alignHorizontal": "left", + "padding": "0 0 0 " + }, + "activeStyle": { + "inheritFrom": "default" + }, + "action": "", + "left": 38, + "top": 0, + "width": 150, + "height": 36, + "text": "Parallel" + } + ] } ] }, @@ -26109,7 +26165,6 @@ "style": { "inheritFrom": "option_toggle_L_center", "font": "Oswald17", - "alignHorizontal": "center", "padding": "0 0 0 6" }, "activeStyle": { @@ -26127,7 +26182,6 @@ "style": { "inheritFrom": "option_toggle_L_center", "font": "Oswald17", - "alignHorizontal": "left", "padding": "0 0 0 6" }, "activeStyle": { @@ -26195,7 +26249,6 @@ "style": { "inheritFrom": "option_toggle_L_center", "font": "Oswald17", - "alignHorizontal": "center", "padding": "0 0 0 6" }, "activeStyle": { @@ -26213,7 +26266,6 @@ "style": { "inheritFrom": "option_toggle_L_center", "font": "Oswald17", - "alignHorizontal": "left", "padding": "0 0 0 6" }, "activeStyle": { @@ -28521,7 +28573,7 @@ "left": 236, "top": 56, "width": 236, - "height": 84, + "height": 176, "itemWidget": { "type": "Container", "style": { @@ -28761,8 +28813,8 @@ "padding": 0 }, "action": "standBy", - "left": 16, - "top": 16, + "left": 10, + "top": 10, "width": 188, "height": 32, "text": "Standby" @@ -28779,8 +28831,8 @@ "padding": 0 }, "action": "reset", - "left": 16, - "top": 48, + "left": 10, + "top": 46, "width": 188, "height": 32, "text": "Soft reset (*RST)" @@ -28797,8 +28849,8 @@ "padding": 0 }, "action": "hard_reset", - "left": 16, - "top": 80, + "left": 10, + "top": 82, "width": 188, "height": 32, "text": "Hard reset" @@ -28815,8 +28867,8 @@ "padding": 0 }, "action": "turnDisplayOff", - "left": 16, - "top": 130, + "left": 10, + "top": 124, "width": 188, "height": 32, "text": "Display off" @@ -28833,8 +28885,8 @@ "inheritFrom": "default", "padding": 0 }, - "left": 16, - "top": 119, + "left": 10, + "top": 118, "width": 188, "height": 2 }, @@ -28850,8 +28902,8 @@ "padding": 0 }, "action": "show_previous_page", - "left": 16, - "top": 162, + "left": 10, + "top": 160, "width": 188, "height": 32, "text": "Cancel" @@ -28859,8 +28911,8 @@ ], "left": 130, "top": 31, - "width": 220, - "height": 210 + "width": 212, + "height": 202 }, { "name": "entering_standby", @@ -48644,6 +48696,322 @@ "width": 168, "height": 484 }, + { + "name": "dcp405b_front_panel", + "style": "", + "widgets": [ + { + "type": "Container", + "style": { + "inheritFrom": "default", + "color": "#f2f2f2", + "backgroundColor": "#ffffff", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "left": 0, + "top": 0, + "width": 168, + "height": 484, + "widgets": [ + { + "type": "Bitmap", + "style": { + "inheritFrom": "default", + "backgroundColor": "#ffffff", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "left": 0, + "top": 0, + "width": 168, + "height": 484, + "bitmap": "DCP405B" + }, + { + "type": "Select", + "style": { + "inheritFrom": "default", + "backgroundColor": "#ffffff", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "data": "simulator.loadState", + "action": "simulator_load", + "left": 4, + "top": 44, + "width": 160, + "height": 28, + "widgets": [ + { + "type": "Text", + "style": { + "inheritFrom": "default", + "font": "Oswald14", + "alignHorizontal": "center", + "color": "#9b9b00", + "backgroundColor": "#ffffff", + "borderSize": 2, + "borderRadius": 4, + "borderColor": "#808080", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "data": "", + "action": "", + "left": 0, + "top": 0, + "width": 160, + "height": 28, + "text": "Load Off" + }, + { + "type": "Text", + "style": { + "inheritFrom": "edit_S", + "font": "Oswald14", + "alignHorizontal": "center", + "color": "#9b9b00", + "backgroundColor": "#ffffff", + "borderSize": 2, + "borderRadius": 4, + "borderColor": "#808080", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "data": "simulator.load", + "action": "", + "left": 0, + "top": 0, + "width": 160, + "height": 28, + "text": "" + } + ] + }, + { + "type": "Select", + "style": { + "inheritFrom": "default", + "backgroundColor": "#ffffff", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "data": "channel.isCC", + "left": 26, + "top": 102, + "width": 20, + "height": 20, + "widgets": [ + { + "type": "Rectangle", + "style": { + "inheritFrom": "default", + "color": "#ffffff", + "backgroundColor": "#ffffff", + "borderSize": 2, + "borderRadius": 8, + "borderColor": "#404040", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "left": 0, + "top": 0, + "width": 20, + "height": 20 + }, + { + "type": "Rectangle", + "style": { + "inheritFrom": "default", + "color": "#ff040b", + "backgroundColor": "#ffffff", + "borderSize": 2, + "borderRadius": 8, + "borderColor": "#404040", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "left": 0, + "top": 0, + "width": 20, + "height": 20 + } + ] + }, + { + "type": "Select", + "style": { + "inheritFrom": "default", + "backgroundColor": "#ffffff", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "data": "channel.isCV", + "left": 50, + "top": 102, + "width": 20, + "height": 20, + "widgets": [ + { + "type": "Rectangle", + "style": { + "inheritFrom": "default", + "color": "#ffffff", + "backgroundColor": "#ffffff", + "borderSize": 2, + "borderRadius": 8, + "borderColor": "#404040", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "left": 0, + "top": 0, + "width": 20, + "height": 20 + }, + { + "type": "Rectangle", + "style": { + "inheritFrom": "default", + "color": "#00ff00", + "backgroundColor": "#ffffff", + "borderSize": 2, + "borderRadius": 8, + "borderColor": "#404040", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "left": 0, + "top": 0, + "width": 20, + "height": 20 + } + ] + }, + { + "type": "Select", + "style": { + "inheritFrom": "default", + "backgroundColor": "#ffffff", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "data": "channel.activeCoupledLed", + "left": 31, + "top": 328, + "width": 20, + "height": 20, + "widgets": [ + { + "type": "Rectangle", + "style": { + "inheritFrom": "default", + "color": "#ffffff", + "backgroundColor": "#ffffff", + "borderSize": 2, + "borderRadius": 8, + "borderColor": "#404040", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "left": 0, + "top": 0, + "width": 20, + "height": 20 + }, + { + "type": "Rectangle", + "style": { + "inheritFrom": "default", + "color": "#00ff00", + "backgroundColor": "#ffffff", + "borderSize": 2, + "borderRadius": 8, + "borderColor": "#404040", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "left": 0, + "top": 0, + "width": 20, + "height": 20 + }, + { + "type": "Rectangle", + "style": { + "inheritFrom": "default", + "color": "#ff040b", + "backgroundColor": "#ffffff", + "borderSize": 2, + "borderRadius": 8, + "borderColor": "#404040", + "padding": 0 + }, + "activeStyle": { + "inheritFrom": "default", + "padding": 0 + }, + "left": 0, + "top": 0, + "width": 20, + "height": 20 + } + ] + } + ] + } + ], + "usedIn": [ + "simulator" + ], + "left": 0, + "top": 0, + "width": 168, + "height": 484 + }, { "name": "dcm220_front_panel", "widgets": [ @@ -50299,16 +50667,6 @@ "borderSize": 0, "padding": 0 }, - { - "name": "mon_dac", - "inheritFrom": "default", - "font": "Oswald24", - "alignHorizontal": "center", - "alignVertical": "center", - "color": "status_warning", - "borderSize": 0, - "padding": 0 - }, { "name": "mon_value", "inheritFrom": "default" @@ -433028,6 +433386,11 @@ "image": "", "bpp": 32 }, + { + "name": "DCP405B", + "image": "", + "bpp": 32 + }, { "name": "DCM220", "image": "", diff --git a/src/eez/apps/psu/calibration.cpp b/src/eez/apps/psu/calibration.cpp index a92f21e2f..8b266a329 100644 --- a/src/eez/apps/psu/calibration.cpp +++ b/src/eez/apps/psu/calibration.cpp @@ -56,14 +56,14 @@ void Value::reset() { max_set = false; min_dac = voltOrCurr - ? g_channel->params->U_CAL_VAL_MIN - : (currentRange == 0 ? g_channel->params->I_CAL_VAL_MIN : g_channel->params->I_CAL_VAL_MIN / 100); + ? g_channel->params.U_CAL_VAL_MIN + : (currentRange == 0 ? g_channel->params.I_CAL_VAL_MIN : g_channel->params.I_CAL_VAL_MIN / 100); mid_dac = voltOrCurr - ? g_channel->params->U_CAL_VAL_MID - : (currentRange == 0 ? g_channel->params->I_CAL_VAL_MID : g_channel->params->I_CAL_VAL_MID / 100); + ? g_channel->params.U_CAL_VAL_MID + : (currentRange == 0 ? g_channel->params.I_CAL_VAL_MID : g_channel->params.I_CAL_VAL_MID / 100); max_dac = voltOrCurr - ? g_channel->params->U_CAL_VAL_MAX - : (currentRange == 0 ? g_channel->params->I_CAL_VAL_MAX : g_channel->params->I_CAL_VAL_MAX / 100); + ? g_channel->params.U_CAL_VAL_MAX + : (currentRange == 0 ? g_channel->params.I_CAL_VAL_MAX : g_channel->params.I_CAL_VAL_MAX / 100); } float Value::getLevelValue() { @@ -91,10 +91,10 @@ void Value::setLevel(int8_t value) { void Value::setLevelValue() { if (voltOrCurr) { g_channel->setVoltage(getLevelValue()); - g_channel->setCurrent(g_channel->params->I_VOLT_CAL); + g_channel->setCurrent(g_channel->params.I_VOLT_CAL); } else { g_channel->setCurrent(getLevelValue()); - g_channel->setVoltage(g_channel->params->U_CURR_CAL); + g_channel->setVoltage(g_channel->params.U_CURR_CAL); } } @@ -117,16 +117,16 @@ void Value::setDacValue(float value) { bool Value::checkRange(float dac, float data, float adc) { float range; if (voltOrCurr) { - range = g_channel->params->U_CAL_VAL_MAX - g_channel->params->U_CAL_VAL_MIN; + range = g_channel->params.U_CAL_VAL_MAX - g_channel->params.U_CAL_VAL_MIN; } else { - range = g_channel->params->I_CAL_VAL_MAX - g_channel->params->I_CAL_VAL_MIN; + range = g_channel->params.I_CAL_VAL_MAX - g_channel->params.I_CAL_VAL_MIN; if (currentRange == 1) { // range /= 5; // 500mA range /= 25; // 50mA } } - float allowedDiff = range * g_channel->params->CALIBRATION_DATA_TOLERANCE_PERCENT / 100; + float allowedDiff = range * g_channel->params.CALIBRATION_DATA_TOLERANCE_PERCENT / 100; float diff; diff = fabsf(dac - data); @@ -135,7 +135,7 @@ bool Value::checkRange(float dac, float data, float adc) { return false; } - if (g_slots[g_channel->slotIndex].moduleType != MODULE_TYPE_DCM220) { + if (g_slots[g_channel->slotIndex].moduleInfo->moduleType != MODULE_TYPE_DCM220) { diff = fabsf(dac - adc); if (diff > allowedDiff) { DebugTrace("ADC check failed: level=%f, adc=%f, diff=%f, allowedDiff=%f", dac, adc, diff, allowedDiff); @@ -165,12 +165,12 @@ void Value::setData(float dac, float data, float adc) { if (min_set && max_set) { if (voltOrCurr) { - minPossible = g_channel->params->U_MIN; - maxPossible = g_channel->params->U_MAX; + minPossible = g_channel->params.U_MIN; + maxPossible = g_channel->params.U_MAX; } else { if (currentRange == 0) { - minPossible = g_channel->params->I_MIN; - maxPossible = g_channel->params->I_MAX; + minPossible = g_channel->params.I_MIN; + maxPossible = g_channel->params.I_MAX; } } DebugTrace("ADC=%f", min_adc); @@ -180,7 +180,7 @@ void Value::setData(float dac, float data, float adc) { bool Value::checkMid() { float mid = remap(mid_dac, min_dac, min_val, max_dac, max_val); - float allowedDiff = g_channel->params->CALIBRATION_MID_TOLERANCE_PERCENT * (max_val - min_val) / 100.0f; + float allowedDiff = g_channel->params.CALIBRATION_MID_TOLERANCE_PERCENT * (max_val - min_val) / 100.0f; float diff = fabsf(mid - mid_val); if (diff <= allowedDiff) { diff --git a/src/eez/apps/psu/channel.cpp b/src/eez/apps/psu/channel.cpp index 9e9532f25..1e969b779 100644 --- a/src/eez/apps/psu/channel.cpp +++ b/src/eez/apps/psu/channel.cpp @@ -44,41 +44,6 @@ namespace psu { //////////////////////////////////////////////////////////////////////////////// -static const char *CH_BOARD_NAMES[] = { "None", "DCP505", "DCP405", "DCP405", "DCM220" }; -static const char *CH_REVISION_NAMES[] = { "None", "R1B3", "R1B1", "R2B5", "R1B1" }; -static const char *CH_BOARD_AND_REVISION_NAMES[] = { "None", "DCP505_R1B3", "DCP405_R1B1", "DCP405_R2B5", "DCM220_R1B1" }; - -static uint16_t CH_BOARD_REVISION_FEATURES[] = { - // CH_BOARD_REVISION_NONE - 0, - - // CH_BOARD_REVISION_DCP505_R1B3 - CH_FEATURE_VOLT | CH_FEATURE_CURRENT | CH_FEATURE_POWER | CH_FEATURE_OE | CH_FEATURE_DPROG | - CH_FEATURE_RPROG | CH_FEATURE_RPOL, - - // CH_BOARD_REVISION_DCP405_R1B1 - CH_FEATURE_VOLT | CH_FEATURE_CURRENT | CH_FEATURE_POWER | CH_FEATURE_OE | CH_FEATURE_DPROG | - CH_FEATURE_RPROG | CH_FEATURE_RPOL, - - // CH_BOARD_REVISION_DCP405_R2B5 - CH_FEATURE_VOLT | CH_FEATURE_CURRENT | CH_FEATURE_POWER | CH_FEATURE_OE | CH_FEATURE_DPROG | - CH_FEATURE_RPROG | CH_FEATURE_RPOL, - - // CH_BOARD_REVISION_DCM220_R1B1 - CH_FEATURE_VOLT | CH_FEATURE_CURRENT | CH_FEATURE_POWER | CH_FEATURE_OE -}; - -static ChannelParams CH_BOARD_REVISION_PARAMS[] = { - // VOLTAGE_GND_OFFSET // CURRENT_GND_OFFSET // CALIBRATION_DATA_TOLERANCE_PERCENT // CALIBRATION_MID_TOLERANCE_PERCENT - { CH_PARAMS_NONE, 0, 0, 10.0f, 1.0f }, // CH_BOARD_REVISION_NONE - { CH_PARAMS_50V_5A, 1.05f, 0.11f, 10.0f, 1.0f }, // CH_BOARD_REVISION_DCP505_R1B3 - { CH_PARAMS_40V_5A, 0.86f, 0.11f, 10.0f, 1.0f }, // CH_BOARD_REVISION_DCP405_R1B1 - { CH_PARAMS_40V_5A, 0.86f, 0.11f, 10.0f, 1.0f }, // CH_BOARD_REVISION_DCP405_R2B5 - { CH_PARAMS_20V_4A, 0, 0, 15.0f, 2.0f } // CH_BOARD_REVISION_DCM220_R1B1 -}; - -//////////////////////////////////////////////////////////////////////////////// - int CH_NUM = 0; Channel Channel::g_channels[CH_MAX]; @@ -217,22 +182,27 @@ float Channel::Simulator::getVoltProgExt() { //////////////////////////////////////////////////////////////////////////////// -void Channel::set(uint8_t slotIndex_, uint8_t subchannelIndex_, uint8_t boardRevision_) { +void Channel::set(uint8_t slotIndex_, uint8_t subchannelIndex_) { + auto slot = g_slots[slotIndex_]; + slotIndex = slotIndex_; - boardRevision = boardRevision_; subchannelIndex = subchannelIndex_; - channelInterface = g_modules[g_slots[slotIndex].moduleType].channelInterfaces ? g_modules[g_slots[slotIndex].moduleType].channelInterfaces[slotIndex] : nullptr; + channelInterface = slot.moduleInfo->channelInterfaces ? slot.moduleInfo->channelInterfaces[slotIndex] : nullptr; + + if (!channelInterface) { + return; + } - params = &CH_BOARD_REVISION_PARAMS[boardRevision]; + channelInterface->getParams(subchannelIndex, params); - u.min = roundChannelValue(UNIT_VOLT, params->U_MIN); - u.max = roundChannelValue(UNIT_VOLT, params->U_MAX); - u.def = roundChannelValue(UNIT_VOLT, params->U_DEF); + u.min = roundChannelValue(UNIT_VOLT, params.U_MIN); + u.max = roundChannelValue(UNIT_VOLT, params.U_MAX); + u.def = roundChannelValue(UNIT_VOLT, params.U_DEF); - i.min = roundChannelValue(UNIT_AMPER, params->I_MIN); - i.max = roundChannelValue(UNIT_AMPER, params->I_MAX); - i.def = roundChannelValue(UNIT_AMPER, params->I_DEF); + i.min = roundChannelValue(UNIT_AMPER, params.I_MIN); + i.max = roundChannelValue(UNIT_AMPER, params.I_MAX); + i.def = roundChannelValue(UNIT_AMPER, params.I_DEF); #ifdef EEZ_PLATFORM_SIMULATOR simulator.load_enabled = true; @@ -383,7 +353,7 @@ void Channel::onPowerDown() { outputEnable(false); doRemoteSensingEnable(false); - if (getFeatures() & CH_FEATURE_RPROG) { + if (params.features & CH_FEATURE_RPROG) { doRemoteProgrammingEnable(false); } @@ -430,7 +400,7 @@ void Channel::reset() { // [SOUR[n]]:VOLT:SENS INTernal doRemoteSensingEnable(false); - if (getFeatures() & CH_FEATURE_RPROG) { + if (params.features & CH_FEATURE_RPROG) { // [SOUR[n]]:VOLT:PROG INTernal doRemoteProgrammingEnable(false); } @@ -448,11 +418,11 @@ void Channel::reset() { // [SOUR[n]]:CURR:STEP // [SOUR[n]]:VOLT // [SOUR[n]]:VOLT:STEP -> set all to default - u.init(params->U_MIN, params->U_DEF_STEP, u.max); - i.init(params->I_MIN, params->I_DEF_STEP, i.max); + u.init(params.U_MIN, params.U_DEF_STEP, u.max); + i.init(params.I_MIN, params.I_DEF_STEP, i.max); maxCurrentLimitCause = MAX_CURRENT_LIMIT_CAUSE_NONE; - p_limit = roundChannelValue(UNIT_WATT, params->PTOT); + p_limit = roundChannelValue(UNIT_WATT, params.PTOT); resetHistory(); @@ -464,8 +434,8 @@ void Channel::reset() { flags.currentTriggerMode = TRIGGER_MODE_FIXED; flags.triggerOutputState = 1; flags.triggerOnListStop = TRIGGER_ON_LIST_STOP_OUTPUT_OFF; - trigger::setVoltage(*this, params->U_MIN); - trigger::setCurrent(*this, params->I_MIN); + trigger::setVoltage(*this, params.U_MIN); + trigger::setCurrent(*this, params.I_MIN); list::resetChannelList(*this); #ifdef EEZ_PLATFORM_SIMULATOR @@ -491,43 +461,43 @@ void Channel::clearCalibrationConf() { cal_conf.flags.i_cal_params_exists_range_high = 0; cal_conf.flags.i_cal_params_exists_range_low = 0; - cal_conf.u.min.dac = cal_conf.u.min.val = cal_conf.u.min.adc = params->U_CAL_VAL_MIN; - cal_conf.u.mid.dac = cal_conf.u.mid.val = cal_conf.u.mid.adc = (params->U_CAL_VAL_MIN + params->U_CAL_VAL_MAX) / 2; - cal_conf.u.max.dac = cal_conf.u.max.val = cal_conf.u.max.adc = params->U_CAL_VAL_MAX; - cal_conf.u.minPossible = params->U_MIN; - cal_conf.u.maxPossible = params->U_MAX; + cal_conf.u.min.dac = cal_conf.u.min.val = cal_conf.u.min.adc = params.U_CAL_VAL_MIN; + cal_conf.u.mid.dac = cal_conf.u.mid.val = cal_conf.u.mid.adc = (params.U_CAL_VAL_MIN + params.U_CAL_VAL_MAX) / 2; + cal_conf.u.max.dac = cal_conf.u.max.val = cal_conf.u.max.adc = params.U_CAL_VAL_MAX; + cal_conf.u.minPossible = params.U_MIN; + cal_conf.u.maxPossible = params.U_MAX; - cal_conf.i[0].min.dac = cal_conf.i[0].min.val = cal_conf.i[0].min.adc = params->I_CAL_VAL_MIN; - cal_conf.i[0].mid.dac = cal_conf.i[0].mid.val = cal_conf.i[0].mid.adc = (params->I_CAL_VAL_MIN + params->I_CAL_VAL_MAX) / 2; - cal_conf.i[0].max.dac = cal_conf.i[0].max.val = cal_conf.i[0].max.adc = params->I_CAL_VAL_MAX; - cal_conf.i[0].minPossible = params->I_MIN; - cal_conf.i[0].maxPossible = params->I_MAX; + cal_conf.i[0].min.dac = cal_conf.i[0].min.val = cal_conf.i[0].min.adc = params.I_CAL_VAL_MIN; + cal_conf.i[0].mid.dac = cal_conf.i[0].mid.val = cal_conf.i[0].mid.adc = (params.I_CAL_VAL_MIN + params.I_CAL_VAL_MAX) / 2; + cal_conf.i[0].max.dac = cal_conf.i[0].max.val = cal_conf.i[0].max.adc = params.I_CAL_VAL_MAX; + cal_conf.i[0].minPossible = params.I_MIN; + cal_conf.i[0].maxPossible = params.I_MAX; - cal_conf.i[1].min.dac = cal_conf.i[1].min.val = cal_conf.i[1].min.adc = params->I_CAL_VAL_MIN / 100; - cal_conf.i[1].mid.dac = cal_conf.i[1].mid.val = cal_conf.i[1].mid.adc = (params->I_CAL_VAL_MIN + params->I_CAL_VAL_MAX) / 2 / 100; - cal_conf.i[1].max.dac = cal_conf.i[1].max.val = cal_conf.i[1].max.adc = params->I_CAL_VAL_MAX / 100; - cal_conf.i[1].minPossible = params->I_MIN; - cal_conf.i[1].maxPossible = params->I_MAX / 100; + cal_conf.i[1].min.dac = cal_conf.i[1].min.val = cal_conf.i[1].min.adc = params.I_CAL_VAL_MIN / 100; + cal_conf.i[1].mid.dac = cal_conf.i[1].mid.val = cal_conf.i[1].mid.adc = (params.I_CAL_VAL_MIN + params.I_CAL_VAL_MAX) / 2 / 100; + cal_conf.i[1].max.dac = cal_conf.i[1].max.val = cal_conf.i[1].max.adc = params.I_CAL_VAL_MAX / 100; + cal_conf.i[1].minPossible = params.I_MIN; + cal_conf.i[1].maxPossible = params.I_MAX / 100; strcpy(cal_conf.calibration_date, ""); strcpy(cal_conf.calibration_remark, CALIBRATION_REMARK_INIT); } void Channel::clearProtectionConf() { - prot_conf.flags.u_state = params->OVP_DEFAULT_STATE; - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { + prot_conf.flags.u_state = params.OVP_DEFAULT_STATE; + if (params.features & CH_FEATURE_HW_OVP) { prot_conf.flags.u_type = 1; // HW } else { prot_conf.flags.u_type = 0; // SW } - prot_conf.flags.i_state = params->OCP_DEFAULT_STATE; - prot_conf.flags.p_state = params->OPP_DEFAULT_STATE; + prot_conf.flags.i_state = params.OCP_DEFAULT_STATE; + prot_conf.flags.p_state = params.OPP_DEFAULT_STATE; - prot_conf.u_delay = params->OVP_DEFAULT_DELAY; + prot_conf.u_delay = params.OVP_DEFAULT_DELAY; prot_conf.u_level = u.max; - prot_conf.i_delay = params->OCP_DEFAULT_DELAY; - prot_conf.p_delay = params->OPP_DEFAULT_DELAY; - prot_conf.p_level = params->OPP_DEFAULT_LEVEL; + prot_conf.i_delay = params.OCP_DEFAULT_DELAY; + prot_conf.p_delay = params.OPP_DEFAULT_DELAY; + prot_conf.p_level = params.OPP_DEFAULT_LEVEL; temperature::sensors[temp_sensor::CH1 + channelIndex].prot_conf.state = OTP_CH_DEFAULT_STATE; temperature::sensors[temp_sensor::CH1 + channelIndex].prot_conf.level = OTP_CH_DEFAULT_LEVEL; @@ -545,7 +515,7 @@ bool Channel::test() { outputEnable(false); doRemoteSensingEnable(false); - if (getFeatures() & CH_FEATURE_RPROG) { + if (params.features & CH_FEATURE_RPROG) { doRemoteProgrammingEnable(false); } @@ -558,7 +528,7 @@ bool Channel::test() { } bool Channel::isInstalled() { - return boardRevision != CH_BOARD_REVISION_NONE; + return channelInterface != nullptr; } bool Channel::isPowerOk() { @@ -591,7 +561,7 @@ void Channel::tick(uint32_t tick_usec) { channelInterface->tick(subchannelIndex, tick_usec); - if (getFeatures() & CH_FEATURE_RPOL) { + if (params.features & CH_FEATURE_RPOL) { unsigned rpol = 0; rpol = channelInterface->getRPol(subchannelIndex); @@ -642,7 +612,7 @@ float Channel::getValuePrecision(Unit unit, float value) const { } float Channel::getVoltageResolution() const { - float precision = params->U_RESOLUTION; // 5 mV; + float precision = params.U_RESOLUTION; // 5 mV; if (calibration::isEnabled()) { precision /= 10; @@ -652,11 +622,11 @@ float Channel::getVoltageResolution() const { } float Channel::getCurrentResolution(float value) const { - float precision = params->I_RESOLUTION; // 0.5mA + float precision = params.I_RESOLUTION; // 0.5mA if (hasSupportForCurrentDualRange()) { if ((!isNaN(value) && value <= 0.05f && isMicroAmperAllowed()) || flags.currentCurrentRange == CURRENT_RANGE_LOW) { - precision = params->I_LOW_RESOLUTION; // 5uA + precision = params.I_LOW_RESOLUTION; // 5uA } } @@ -668,7 +638,7 @@ float Channel::getCurrentResolution(float value) const { } float Channel::getPowerResolution() const { - return params->P_RESOLUTION; // 1 mW; + return params.P_RESOLUTION; // 1 mW; } bool Channel::isMicroAmperAllowed() const { @@ -869,7 +839,7 @@ void Channel::update() { setCurrent(i.set); doOutputEnable(flags.outputEnabled); doRemoteSensingEnable(flags.senseEnabled); - if (getFeatures() & CH_FEATURE_RPROG) { + if (params.features & CH_FEATURE_RPROG) { doRemoteProgrammingEnable(flags.rprogEnabled); } @@ -892,37 +862,37 @@ void Channel::doCalibrationEnable(bool enable) { flags._calEnabled = enable; if (enable) { - u.min = roundChannelValue(UNIT_VOLT, MAX(cal_conf.u.minPossible, params->U_MIN)); + u.min = roundChannelValue(UNIT_VOLT, MAX(cal_conf.u.minPossible, params.U_MIN)); if (u.limit < u.min) u.limit = u.min; if (u.set < u.min) setVoltage(u.min); - u.max = roundChannelValue(UNIT_VOLT, MIN(cal_conf.u.maxPossible, params->U_MAX)); + u.max = roundChannelValue(UNIT_VOLT, MIN(cal_conf.u.maxPossible, params.U_MAX)); if (u.limit > u.max) u.limit = u.max; if (u.set > u.max) setVoltage(u.max); - i.min = roundChannelValue(UNIT_AMPER, MAX(cal_conf.i[0].minPossible, params->I_MIN)); - if (i.min < params->I_MIN) - i.min = params->I_MIN; + i.min = roundChannelValue(UNIT_AMPER, MAX(cal_conf.i[0].minPossible, params.I_MIN)); + if (i.min < params.I_MIN) + i.min = params.I_MIN; if (i.limit < i.min) i.limit = i.min; if (i.set < i.min) setCurrent(i.min); - i.max = roundChannelValue(UNIT_AMPER, MIN(cal_conf.i[0].maxPossible, params->I_MAX)); + i.max = roundChannelValue(UNIT_AMPER, MIN(cal_conf.i[0].maxPossible, params.I_MAX)); if (i.limit > i.max) i.limit = i.max; if (i.set > i.max) setCurrent(i.max); } else { - u.min = roundChannelValue(UNIT_VOLT, params->U_MIN); - u.max = roundChannelValue(UNIT_VOLT, params->U_MAX); + u.min = roundChannelValue(UNIT_VOLT, params.U_MIN); + u.max = roundChannelValue(UNIT_VOLT, params.U_MAX); - i.min = roundChannelValue(UNIT_AMPER, params->I_MIN); - i.max = roundChannelValue(UNIT_AMPER, params->I_MAX); + i.min = roundChannelValue(UNIT_AMPER, params.I_MIN); + i.max = roundChannelValue(UNIT_AMPER, params.I_MAX); } u.def = roundChannelValue(UNIT_VOLT, u.min); @@ -993,7 +963,7 @@ float Channel::getCalibratedVoltage(float value) { } #if !defined(EEZ_PLATFORM_SIMULATOR) - value += params->VOLTAGE_GND_OFFSET; + value += params.VOLTAGE_GND_OFFSET; #endif return value; @@ -1007,8 +977,8 @@ void Channel::doSetVoltage(float value) { prot_conf.u_level = u.set; } - if (params->U_MAX != params->U_MAX_CONF) { - value = remap(value, 0, 0, params->U_MAX_CONF, params->U_MAX); + if (params.U_MAX != params.U_MAX_CONF) { + value = remap(value, 0, 0, params.U_MAX_CONF, params.U_MAX); } value = getCalibratedVoltage(value); @@ -1040,8 +1010,8 @@ void Channel::doSetCurrent(float value) { i.set = value; i.mon_dac = 0; - if (params->I_MAX != params->I_MAX_CONF) { - value = remap(value, 0, 0, params->I_MAX_CONF, params->I_MAX); + if (params.I_MAX != params.I_MAX_CONF) { + value = remap(value, 0, 0, params.I_MAX_CONF, params.I_MAX); } if (isCurrentCalibrationEnabled()) { @@ -1130,22 +1100,6 @@ const char *Channel::getCvModeStr() { return "UR"; } -const char *Channel::getBoardName() { - return CH_BOARD_NAMES[boardRevision]; -} - -const char *Channel::getRevisionName() { - return CH_REVISION_NAMES[boardRevision]; -} - -const char *Channel::getBoardAndRevisionName() { - return CH_BOARD_AND_REVISION_NAMES[boardRevision]; -} - -uint16_t Channel::getFeatures() { - return CH_BOARD_REVISION_FEATURES[boardRevision]; -} - float Channel::getVoltageLimit() const { return u.limit; } @@ -1227,7 +1181,7 @@ float Channel::getPowerLimit() const { } float Channel::getPowerMaxLimit() const { - return params->PTOT; + return params.PTOT; } void Channel::setPowerLimit(float limit) { @@ -1306,13 +1260,12 @@ float Channel::getDualRangeGndOffset() { #ifdef EEZ_PLATFORM_SIMULATOR return 0; #else - return flags.currentCurrentRange == CURRENT_RANGE_LOW ? (params->CURRENT_GND_OFFSET / 100) : params->CURRENT_GND_OFFSET; + return flags.currentCurrentRange == CURRENT_RANGE_LOW ? (params.CURRENT_GND_OFFSET / 100) : params.CURRENT_GND_OFFSET; #endif } bool Channel::hasSupportForCurrentDualRange() const { - return (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) && - (boardRevision == CH_BOARD_REVISION_DCP405_R1B1 || boardRevision == CH_BOARD_REVISION_DCP405_R2B5); + return params.features & CH_FEATURE_CURRENT_DUAL_RANGE ? true : false; } void Channel::setCurrentRangeSelectionMode(CurrentRangeSelectionMode mode) { @@ -1342,7 +1295,7 @@ void Channel::enableAutoSelectCurrentRange(bool enable) { } float Channel::getDualRangeMax() { - return flags.currentCurrentRange == CURRENT_RANGE_LOW ? (params->I_MAX / 100) : params->I_MAX; + return flags.currentCurrentRange == CURRENT_RANGE_LOW ? (params.I_MAX / 100) : params.I_MAX; } void Channel::setCurrentRange(uint8_t currentCurrentRange) { diff --git a/src/eez/apps/psu/channel.h b/src/eez/apps/psu/channel.h index 2537c187a..b7a7b6a9f 100644 --- a/src/eez/apps/psu/channel.h +++ b/src/eez/apps/psu/channel.h @@ -44,16 +44,6 @@ struct Value; enum DisplayValue { DISPLAY_VALUE_VOLTAGE, DISPLAY_VALUE_CURRENT, DISPLAY_VALUE_POWER }; -enum ChannelFeatures { - CH_FEATURE_VOLT = (1 << 1), - CH_FEATURE_CURRENT = (1 << 2), - CH_FEATURE_POWER = (1 << 3), - CH_FEATURE_OE = (1 << 4), - CH_FEATURE_DPROG = (1 << 5), - CH_FEATURE_RPROG = (1 << 7), - CH_FEATURE_RPOL = (1 << 8) -}; - enum TriggerMode { TRIGGER_MODE_FIXED, TRIGGER_MODE_LIST, TRIGGER_MODE_STEP }; enum TriggerOnListStop { @@ -88,65 +78,6 @@ struct ProtectionValue { uint32_t alarm_started; }; -struct ChannelParams { - float U_MIN; - float U_DEF; - float U_MAX; - float U_MAX_CONF; - float U_MIN_STEP; - float U_DEF_STEP; - float U_MAX_STEP; - float U_CAL_VAL_MIN; - float U_CAL_VAL_MID; - float U_CAL_VAL_MAX; - float U_CURR_CAL; - bool OVP_DEFAULT_STATE; - float OVP_MIN_DELAY; - float OVP_DEFAULT_DELAY; - float OVP_MAX_DELAY; - float I_MIN; - float I_DEF; - float I_MAX; - float I_MAX_CONF; - float I_MIN_STEP; - float I_DEF_STEP; - float I_MAX_STEP; - float I_CAL_VAL_MIN; - float I_CAL_VAL_MID; - float I_CAL_VAL_MAX; - float I_VOLT_CAL; - bool OCP_DEFAULT_STATE; - float OCP_MIN_DELAY; - float OCP_DEFAULT_DELAY; - float OCP_MAX_DELAY; - bool OPP_DEFAULT_STATE; - float OPP_MIN_DELAY; - float OPP_DEFAULT_DELAY; - float OPP_MAX_DELAY; - float OPP_MIN_LEVEL; - float OPP_DEFAULT_LEVEL; - float OPP_MAX_LEVEL; - float SOA_VIN; - float SOA_PREG_CURR; - float SOA_POSTREG_PTOT; - float PTOT; - float U_RESOLUTION; - float I_RESOLUTION; - float I_LOW_RESOLUTION; - float P_RESOLUTION; - - float VOLTAGE_GND_OFFSET; // [V], (1375 / 65535) * (40V | 50V) - float CURRENT_GND_OFFSET; // [A] - - /// Maximum difference, in percentage, between ADC - /// and real value during calibration. - float CALIBRATION_DATA_TOLERANCE_PERCENT; - - /// Maximum difference, in percentage, between calculated mid value - /// and real mid value during calibration. - float CALIBRATION_MID_TOLERANCE_PERCENT; -}; - /// PSU channel. class Channel { friend class DigitalAnalogConverter; @@ -348,8 +279,6 @@ class Channel { // Slot index. Starts from 0. uint8_t slotIndex; - uint8_t boardRevision; - /// Channel index. Starts from 0. uint8_t channelIndex; @@ -358,7 +287,7 @@ class Channel { ChannelInterface *channelInterface; - ChannelParams *params; + ChannelParams params; Flags flags; @@ -380,7 +309,7 @@ class Channel { Simulator simulator; #endif // EEZ_PLATFORM_SIMULATOR - void set(uint8_t slotIndex, uint8_t subchannelIndex, uint8_t boardRevision); + void set(uint8_t slotIndex, uint8_t subchannelIndex); void setChannelIndex(uint8_t channelIndex); @@ -508,18 +437,6 @@ class Channel { /// Returns "CC", "CV" or "UR" const char *getCvModeStr(); - /// Returns name of the board of this channel. - const char *getBoardName(); - - /// Returns name of the revison of this channel. - const char *getRevisionName(); - - /// Returns name of the board and revison of this channel. - const char *getBoardAndRevisionName(); - - /// Returns features present (check ChannelFeatures) in board revision of this channel. - uint16_t getFeatures(); - /// Returns currently set voltage limit float getVoltageLimit() const; diff --git a/src/eez/apps/psu/channel_dispatcher.cpp b/src/eez/apps/psu/channel_dispatcher.cpp index a8fc525fd..94ab77f82 100644 --- a/src/eez/apps/psu/channel_dispatcher.cpp +++ b/src/eez/apps/psu/channel_dispatcher.cpp @@ -85,14 +85,7 @@ bool isCouplingTypeAllowed(CouplingType couplingType, int *err) { } if (couplingType == COUPLING_TYPE_PARALLEL || couplingType == COUPLING_TYPE_SERIES) { - if (g_slots[0].moduleType != g_slots[1].moduleType) { - if (err) { - *err = SCPI_ERROR_HARDWARE_MISSING; - } - return false; - } - - if (g_slots[0].moduleType != MODULE_TYPE_DCP405 && g_slots[0].moduleType != MODULE_TYPE_DCP505) { + if (!(Channel::get(0).params.features & CH_FEATURE_COUPLING) || !(Channel::get(1).params.features & CH_FEATURE_COUPLING)) { if (err) { *err = SCPI_ERROR_HARDWARE_MISSING; } @@ -116,7 +109,7 @@ bool setCouplingType(CouplingType couplingType, int *err) { channel.outputEnable(false); channel.remoteSensingEnable(false); - if (channel.getFeatures() & CH_FEATURE_RPROG) { + if (channel.params.features & CH_FEATURE_RPROG) { channel.remoteProgrammingEnable(false); } @@ -825,9 +818,9 @@ float getPowerMinLimit(const Channel &channel) { float getPowerMaxLimit(const Channel &channel) { if (channel.channelIndex < 2 && (g_couplingType == COUPLING_TYPE_SERIES || g_couplingType == COUPLING_TYPE_PARALLEL)) { - return 2 * MIN(Channel::get(0).params->PTOT, Channel::get(1).params->PTOT); + return 2 * MIN(Channel::get(0).params.PTOT, Channel::get(1).params.PTOT); } - return channel.params->PTOT; + return channel.params.PTOT; } float getPowerDefaultLimit(const Channel &channel) { @@ -867,23 +860,23 @@ float getOppLevel(Channel &channel) { float getOppMinLevel(Channel &channel) { if (channel.channelIndex < 2 && (g_couplingType == COUPLING_TYPE_SERIES || g_couplingType == COUPLING_TYPE_PARALLEL)) { - return 2 * MAX(Channel::get(0).params->OPP_MIN_LEVEL, Channel::get(1).params->OPP_MIN_LEVEL); + return 2 * MAX(Channel::get(0).params.OPP_MIN_LEVEL, Channel::get(1).params.OPP_MIN_LEVEL); } - return channel.params->OPP_MIN_LEVEL; + return channel.params.OPP_MIN_LEVEL; } float getOppMaxLevel(Channel &channel) { if (channel.channelIndex < 2 && (g_couplingType == COUPLING_TYPE_SERIES || g_couplingType == COUPLING_TYPE_PARALLEL)) { - return 2 * MIN(Channel::get(0).params->OPP_MAX_LEVEL, Channel::get(1).params->OPP_MAX_LEVEL); + return 2 * MIN(Channel::get(0).params.OPP_MAX_LEVEL, Channel::get(1).params.OPP_MAX_LEVEL); } - return channel.params->OPP_MAX_LEVEL; + return channel.params.OPP_MAX_LEVEL; } float getOppDefaultLevel(Channel &channel) { if (channel.channelIndex < 2 && (g_couplingType == COUPLING_TYPE_SERIES || g_couplingType == COUPLING_TYPE_PARALLEL)) { - return Channel::get(0).params->OPP_DEFAULT_LEVEL + Channel::get(1).params->OPP_DEFAULT_LEVEL; + return Channel::get(0).params.OPP_DEFAULT_LEVEL + Channel::get(1).params.OPP_DEFAULT_LEVEL; } - return channel.params->OPP_DEFAULT_LEVEL; + return channel.params.OPP_DEFAULT_LEVEL; } void setOppParameters(Channel &channel, int state, float level, float delay) { diff --git a/src/eez/apps/psu/conf.h b/src/eez/apps/psu/conf.h index 10160e41d..a4b3d55c4 100644 --- a/src/eez/apps/psu/conf.h +++ b/src/eez/apps/psu/conf.h @@ -46,7 +46,7 @@ option. #define MIN_POWER_UP_DELAY 5 /// Default calibration password. -#define CALIBRATION_PASSWORD_DEFAULT "eezpsu" +#define CALIBRATION_PASSWORD_DEFAULT "eezbb3" /// Is OTP enabled by default? #define OTP_AUX_DEFAULT_STATE 1 diff --git a/src/eez/apps/psu/conf_channel.h b/src/eez/apps/psu/conf_channel.h deleted file mode 100644 index 4d8a0ae47..000000000 --- a/src/eez/apps/psu/conf_channel.h +++ /dev/null @@ -1,60 +0,0 @@ -/* - * EEZ Modular Firmware - * Copyright (C) 2015-present, Envox d.o.o. - * - * 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 3 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, see . - */ - -#pragma once - - -#define CH_BOARD_REVISION_NONE 0 - -#define CH_BOARD_REVISION_DCP505_R1B3 1 - -#define CH_BOARD_REVISION_DCP405_R1B1 2 -#define CH_BOARD_REVISION_DCP405_R2B5 3 - -#define CH_BOARD_REVISION_DCM220_R1B1 4 - - -// OVP_DEFAULT_STATE, OVP_MIN_DELAY, OVP_DEFAULT_DELAY, OVP_MAX_DELAY -#define CH_PARAMS_OVP false, 0.0f, 0.005f, 10.0f - -// U_MIN, U_DEF, U_MAX, U_MAX_CONF, U_MIN_STEP, U_DEF_STEP, U_MAX_STEP, U_CAL_VAL_MIN, U_CAL_VAL_MID, U_CAL_VAL_MAX, U_CURR_CAL -#define CH_PARAMS_U_50V 0.0f, 0.0f, 50.0f, 50.0f, 0.01f, 0.1f, 5.0f, 0.15f, 24.1f, 48.0f, 25.0f, CH_PARAMS_OVP -// U_MIN, U_DEF, U_MAX, U_MAX_CONF, U_MIN_STEP, U_DEF_STEP, U_MAX_STEP, U_CAL_VAL_MIN, U_CAL_VAL_MID, U_CAL_VAL_MAX, U_CURR_CAL -#define CH_PARAMS_U_40V 0.0f, 0.0f, 40.0f, 40.0f, 0.01f, 0.1f, 5.0f, 0.15f, 20.0f, 38.0f, 20.0f, CH_PARAMS_OVP -// U_MIN, U_DEF, U_MAX, U_MAX_CONF, U_MIN_STEP, U_DEF_STEP, U_MAX_STEP, U_CAL_VAL_MIN, U_CAL_VAL_MID, U_CAL_VAL_MAX, U_CURR_CAL -#define CH_PARAMS_U_20V 0.0f, 0.0f, 20.0f, 20.0f, 0.01f, 0.1f, 5.0f, 0.75f, 10.0f, 18.0f, 20.0f, CH_PARAMS_OVP - -// OCP_DEFAULT_STATE, OCP_MIN_DELAY, OCP_DEFAULT_DELAY, OCP_MAX_DELAY -#define CH_PARAMS_OCP false, 0.0f, 0.02f, 10.0f - -// I_MIN, I_DEF, I_MAX, I_MAX_CONF, I_MIN_STEP, I_DEF_STEP, I_MAX_STEP, I_CAL_VAL_MIN, I_CAL_VAL_MID, I_CAL_VAL_MAX, I_VOLT_CAL -#define CH_PARAMS_I_5A 0.0f, 0.0f, 5.0f, 5.0f, 0.01f, 0.01f, 1.0f, 0.05f, 2.425f, 4.8f, 0.1f, CH_PARAMS_OCP -// I_MIN, I_DEF, I_MAX, I_MAX_CONF, I_MIN_STEP, I_DEF_STEP, I_MAX_STEP, I_CAL_VAL_MIN, I_CAL_VAL_MID, I_CAL_VAL_MAX, I_VOLT_CAL -#define CH_PARAMS_I_4A 0.0f, 0.0f, 4.0f, 4.0f, 0.01f, 0.01f, 1.0f, 0.05f, 1.95f, 3.8f, 0.5f, CH_PARAMS_OCP - -// OPP_MIN_DELAY, OPP_DEFAULT_DELAY, OPP_MAX_DELAY -#define CH_PARAMS_OPP_DELAY 1.0f, 10.0f, 300.0f - -// OPP_DEFAULT_STATE, OPP_MIN_LEVEL, SOA_VIN, OPP_DEFAULT_LEVEL, SOA_PREG_CURR, OPP_MAX_LEVEL, SOA_POSTREG_PTOT, PTOT, U_RESOLUTION, I_RESOLUTION, I_LOW_RESOLUTION, P_RESOLUTION -#define CH_PARAMS_NONE CH_PARAMS_U_50V, CH_PARAMS_I_5A, true, CH_PARAMS_OPP_DELAY, 0.0f, 160.0f, 155.0f, 58.0f, 5.0f, 25.0f, 155.0f, 0, 0, 0, 0 -// OPP_DEFAULT_STATE, OPP_MIN_LEVEL, SOA_VIN, OPP_DEFAULT_LEVEL, SOA_PREG_CURR, OPP_MAX_LEVEL, SOA_POSTREG_PTOT, PTOT, U_RESOLUTION, I_RESOLUTION, I_LOW_RESOLUTION, P_RESOLUTION -#define CH_PARAMS_50V_5A CH_PARAMS_U_50V, CH_PARAMS_I_5A, true, CH_PARAMS_OPP_DELAY, 0.0f, 160.0f, 155.0f, 58.0f, 5.0f, 25.0f, 155.0f, 0.005f, 0.0005f, 0.000005f, 0.001f -// OPP_DEFAULT_STATE, OPP_MIN_LEVEL, SOA_VIN, OPP_DEFAULT_LEVEL, SOA_PREG_CURR, OPP_MAX_LEVEL, SOA_POSTREG_PTOT, PTOT, U_RESOLUTION, I_RESOLUTION, I_LOW_RESOLUTION, P_RESOLUTION -#define CH_PARAMS_40V_5A CH_PARAMS_U_40V, CH_PARAMS_I_5A, true, CH_PARAMS_OPP_DELAY, 0.0f, 160.0f, 155.0f, 58.0f, 5.0f, 25.0f, 155.0f, 0.005f, 0.0005f, 0.000005f, 0.001f -// OPP_DEFAULT_STATE, OPP_MIN_LEVEL, SOA_VIN, OPP_DEFAULT_LEVEL, SOA_PREG_CURR, OPP_MAX_LEVEL, SOA_POSTREG_PTOT, PTOT, U_RESOLUTION, I_RESOLUTION, I_LOW_RESOLUTION, P_RESOLUTION -#define CH_PARAMS_20V_4A CH_PARAMS_U_20V, CH_PARAMS_I_4A, true, CH_PARAMS_OPP_DELAY, 0.0f, 70.0f, 80.0f, 58.0f, 5.0f, 25.0f, 80.0f, 0.01f, 0.02f, 0, 0.001f diff --git a/src/eez/apps/psu/gui/calibration.cpp b/src/eez/apps/psu/gui/calibration.cpp index 7766e7b6f..96e5c964a 100644 --- a/src/eez/apps/psu/gui/calibration.cpp +++ b/src/eez/apps/psu/gui/calibration.cpp @@ -57,7 +57,7 @@ void onStartPasswordOk() { g_channel->prot_conf.flags.i_state = 0; g_channel->prot_conf.flags.p_state = 0; - if (g_channel->getFeatures() & CH_FEATURE_RPROG) { + if (g_channel->params.features & CH_FEATURE_RPROG) { g_channel->remoteProgrammingEnable(false); } @@ -288,9 +288,9 @@ void finishStop() { g_channel->outputEnable(false); - g_channel->prot_conf.flags.u_state = g_channel->params->OVP_DEFAULT_STATE; - g_channel->prot_conf.flags.i_state = g_channel->params->OCP_DEFAULT_STATE; - g_channel->prot_conf.flags.p_state = g_channel->params->OPP_DEFAULT_STATE; + g_channel->prot_conf.flags.u_state = g_channel->params.OVP_DEFAULT_STATE; + g_channel->prot_conf.flags.i_state = g_channel->params.OCP_DEFAULT_STATE; + g_channel->prot_conf.flags.p_state = g_channel->params.OPP_DEFAULT_STATE; (*g_stopCallback)(); } diff --git a/src/eez/apps/psu/gui/data.cpp b/src/eez/apps/psu/gui/data.cpp index 2e2e497eb..c7b9a93a2 100644 --- a/src/eez/apps/psu/gui/data.cpp +++ b/src/eez/apps/psu/gui/data.cpp @@ -173,6 +173,7 @@ EnumItem g_dstRuleEnumDefinition[] = { { datetime::DST_RULE_OFF, "Off" }, #if defined(EEZ_PLATFORM_SIMULATOR) EnumItem g_moduleTypeEnumDefinition[] = { { MODULE_TYPE_NONE, "None" }, { MODULE_TYPE_DCP405, "DCP405" }, + { MODULE_TYPE_DCP405B, "DCP405B" }, { MODULE_TYPE_DCM220, "DCM220" }, { 0, 0 } }; #endif @@ -319,7 +320,7 @@ bool compare_CHANNEL_BOARD_INFO_LABEL_value(const Value &a, const Value &b) { } void CHANNEL_BOARD_INFO_LABEL_value_to_text(const Value &value, char *text, int count) { - snprintf(text, count - 1, "CH%d board:", value.getInt()); + snprintf(text, count - 1, "CH%d board:", value.getInt() + 1); text[count - 1] = 0; } @@ -602,7 +603,7 @@ void CHANNEL_TITLE_value_to_text(const Value &value, char *text, int count) { if (channel.flags.trackingEnabled) { snprintf(text, count - 1, "\xA2 #%d", channel.channelIndex + 1); } else { - snprintf(text, count - 1, "%s #%d", channel.getBoardName(), channel.channelIndex + 1); + snprintf(text, count - 1, "%s #%d", g_slots[channel.slotIndex].moduleInfo->moduleName, channel.channelIndex + 1); } } @@ -637,13 +638,16 @@ bool compare_CHANNEL_LONG_TITLE_value(const Value &a, const Value &b) { } void CHANNEL_LONG_TITLE_value_to_text(const Value &value, char *text, int count) { - Channel &channel = Channel::get(value.getInt()); + auto &channel = Channel::get(value.getInt()); + auto &slot = g_slots[channel.slotIndex]; if (channel.flags.trackingEnabled) { - snprintf(text, count - 1, "\xA2 %s #%d: %dV/%dA, %s", channel.getBoardName(), channel.channelIndex + 1, - (int)floor(channel.params->U_MAX), (int)floor(channel.params->I_MAX), channel.getRevisionName()); + snprintf(text, count - 1, "\xA2 %s #%d: %dV/%dA, R%dB%d", slot.moduleInfo->moduleName, slot.channelIndex + 1, + (int)floor(channel.params.U_MAX), (int)floor(channel.params.I_MAX), + (int)(slot.moduleRevision >> 8), (int)(slot.moduleRevision & 0xFF)); } else { - snprintf(text, count - 1, "%s #%d: %dV/%dA, %s", channel.getBoardName(), channel.channelIndex + 1, - (int)floor(channel.params->U_MAX), (int)floor(channel.params->I_MAX), channel.getRevisionName()); + snprintf(text, count - 1, "%s #%d: %dV/%dA, R%dB%d", g_slots[channel.slotIndex].moduleInfo->moduleName, slot.channelIndex + 1, + (int)floor(channel.params.U_MAX), (int)floor(channel.params.I_MAX), + (int)(slot.moduleRevision >> 8), (int)(slot.moduleRevision & 0xFF)); } } @@ -1126,7 +1130,7 @@ int getDefaultView(int channelIndex) { Channel &channel = Channel::get(channelIndex); if (channel.isInstalled()) { if (channel.isOk()) { - int numChannels = g_modules[g_slots[channel.slotIndex].moduleType].numChannels; + int numChannels = g_slots[channel.slotIndex].moduleInfo->numChannels; if (numChannels == 1) { if (channel_dispatcher::getCouplingType() == channel_dispatcher::COUPLING_TYPE_SERIES && channel.channelIndex == 1) { if (persist_conf::devConf.flags.channelsViewMode == CHANNELS_VIEW_MODE_NUMERIC || persist_conf::devConf.flags.channelsViewMode == CHANNELS_VIEW_MODE_VERT_BAR) { @@ -1215,7 +1219,7 @@ void data_slot_max_view(data::DataOperationEnum operation, data::Cursor &cursor, Channel &channel = Channel::get(cursor.i); if (channel.isInstalled()) { if (channel.isOk()) { - int numChannels = g_modules[g_slots[channel.slotIndex].moduleType].numChannels; + int numChannels = g_slots[channel.slotIndex].moduleInfo->numChannels; if (numChannels == 1) { if (persist_conf::devConf.flags.channelsViewModeInMax == CHANNELS_VIEW_MODE_IN_MAX_NUMERIC) { value = channel.isOutputEnabled() ? PAGE_ID_SLOT_MAX_1CH_NUM_ON : PAGE_ID_SLOT_MAX_1CH_NUM_OFF; @@ -1244,7 +1248,7 @@ int getMinView(int channelIndex) { Channel &channel = Channel::get(channelIndex); if (channel.isInstalled()) { if (channel.isOk()) { - int numChannels = g_modules[g_slots[channel.slotIndex].moduleType].numChannels; + int numChannels = g_slots[channel.slotIndex].moduleInfo->numChannels; if (numChannels == 1) { if (channel_dispatcher::getCouplingType() == channel_dispatcher::COUPLING_TYPE_SERIES && channel.channelIndex == 1) { return PAGE_ID_SLOT_MIN_1CH_COUPLED_SERIES; @@ -1296,7 +1300,7 @@ int getMicroView(int channelIndex) { Channel &channel = Channel::get(channelIndex); if (channel.isInstalled()) { if (channel.isOk()) { - int numChannels = g_modules[g_slots[channel.slotIndex].moduleType].numChannels; + int numChannels = g_slots[channel.slotIndex].moduleInfo->numChannels; if (numChannels == 1) { if (channel_dispatcher::getCouplingType() == channel_dispatcher::COUPLING_TYPE_SERIES && channel.channelIndex == 1) { return PAGE_ID_SLOT_MICRO_1CH_COUPLED_SERIES; @@ -1974,7 +1978,7 @@ void data_channel_protection_ovp_type(data::DataOperationEnum operation, data::C if (operation == data::DATA_OPERATION_GET) { int iChannel = cursor.i >= 0 ? cursor.i : (g_channel ? g_channel->channelIndex : 0); Channel &channel = Channel::get(iChannel); - if (g_slots[channel.slotIndex].moduleType == MODULE_TYPE_DCP405) { + if (channel.params.features & CH_FEATURE_HW_OVP) { ChSettingsProtectionSetPage *page = (ChSettingsProtectionSetPage *)getPage(PAGE_ID_CH_SETTINGS_PROT_OVP); if (page) { value = page->type ? 0 : 1; @@ -2012,9 +2016,9 @@ void data_channel_protection_ovp_delay(data::DataOperationEnum operation, data:: } } } else if (operation == data::DATA_OPERATION_GET_MIN) { - value = MakeValue(channel.params->OVP_MIN_DELAY, UNIT_SECOND); + value = MakeValue(channel.params.OVP_MIN_DELAY, UNIT_SECOND); } else if (operation == data::DATA_OPERATION_GET_MAX) { - value = MakeValue(channel.params->OVP_MAX_DELAY, UNIT_SECOND); + value = MakeValue(channel.params.OVP_MAX_DELAY, UNIT_SECOND); } else if (operation == data::DATA_OPERATION_SET) { channel_dispatcher::setOvpParameters(channel, channel.prot_conf.flags.u_state ? 1 : 0, channel.prot_conf.flags.u_type ? 1 : 0, channel_dispatcher::getUProtectionLevel(channel), value.getFloat()); } @@ -2069,9 +2073,9 @@ void data_channel_protection_ocp_delay(data::DataOperationEnum operation, data:: } } } else if (operation == data::DATA_OPERATION_GET_MIN) { - value = MakeValue(channel.params->OCP_MIN_DELAY, UNIT_SECOND); + value = MakeValue(channel.params.OCP_MIN_DELAY, UNIT_SECOND); } else if (operation == data::DATA_OPERATION_GET_MAX) { - value = MakeValue(channel.params->OCP_MAX_DELAY, UNIT_SECOND); + value = MakeValue(channel.params.OCP_MAX_DELAY, UNIT_SECOND); } else if (operation == data::DATA_OPERATION_SET) { channel_dispatcher::setOcpParameters(channel, channel.prot_conf.flags.i_state ? 1 : 0, value.getFloat()); } @@ -2156,9 +2160,9 @@ void data_channel_protection_opp_delay(data::DataOperationEnum operation, data:: } } } else if (operation == data::DATA_OPERATION_GET_MIN) { - value = MakeValue(channel.params->OPP_MIN_DELAY, UNIT_SECOND); + value = MakeValue(channel.params.OPP_MIN_DELAY, UNIT_SECOND); } else if (operation == data::DATA_OPERATION_GET_MAX) { - value = MakeValue(channel.params->OPP_MAX_DELAY, UNIT_SECOND); + value = MakeValue(channel.params.OPP_MAX_DELAY, UNIT_SECOND); } else if (operation == data::DATA_OPERATION_SET) { channel_dispatcher::setOppParameters(channel, channel.prot_conf.flags.p_state ? 1 : 0, channel_dispatcher::getPowerProtectionLevel(channel), value.getFloat()); } @@ -2326,7 +2330,7 @@ void data_channel_has_advanced_options(data::DataOperationEnum operation, data:: if (operation == data::DATA_OPERATION_GET) { int iChannel = cursor.i >= 0 ? cursor.i : (g_channel ? g_channel->channelIndex : 0); Channel &channel = Channel::get(iChannel); - value = g_slots[channel.slotIndex].moduleType == MODULE_TYPE_DCP405 ? 1 : 0; + value = channel.params.features & (CH_FEATURE_DPROG | CH_FEATURE_RPROG | CH_FEATURE_RPOL) ? 1 : 0; } } @@ -2340,7 +2344,7 @@ void data_channel_rsense_status(data::DataOperationEnum operation, data::Cursor void data_channel_rprog_installed(data::DataOperationEnum operation, data::Cursor &cursor, data::Value &value) { if (operation == data::DATA_OPERATION_GET) { - value = g_channel->getFeatures() & CH_FEATURE_RPROG ? 1 : 0; + value = g_channel->params.features & CH_FEATURE_RPROG ? 1 : 0; } } @@ -2356,7 +2360,7 @@ void data_channel_dprog_installed(data::DataOperationEnum operation, data::Curso if (operation == data::DATA_OPERATION_GET) { int iChannel = cursor.i >= 0 ? cursor.i : (g_channel ? g_channel->channelIndex : 0); Channel &channel = Channel::get(iChannel); - value = channel.getFeatures() & CH_FEATURE_DPROG ? 1 : 0; + value = channel.params.features & CH_FEATURE_DPROG ? 1 : 0; } } @@ -2710,7 +2714,7 @@ void data_sys_fan_speed(data::DataOperationEnum operation, data::Cursor &cursor, void data_channel_board_info_label(data::DataOperationEnum operation, data::Cursor &cursor, data::Value &value) { if (operation == data::DATA_OPERATION_GET) { if (cursor.i >= 0 && cursor.i < CH_NUM) { - value = data::Value(cursor.i + 1, VALUE_TYPE_CHANNEL_BOARD_INFO_LABEL); + value = data::Value(cursor.i, VALUE_TYPE_CHANNEL_BOARD_INFO_LABEL); } } } @@ -2718,7 +2722,7 @@ void data_channel_board_info_label(data::DataOperationEnum operation, data::Curs void data_channel_board_info_revision(data::DataOperationEnum operation, data::Cursor &cursor, data::Value &value) { if (operation == data::DATA_OPERATION_GET) { if (cursor.i >= 0 && cursor.i < CH_NUM) { - value = data::Value(Channel::get(cursor.i).getBoardAndRevisionName()); + value = Value((int)Channel::get(cursor.i).slotIndex, VALUE_TYPE_SLOT_INFO); } } } diff --git a/src/eez/apps/psu/gui/page_ch_settings_adv.cpp b/src/eez/apps/psu/gui/page_ch_settings_adv.cpp index 90671c482..89c70fd3c 100644 --- a/src/eez/apps/psu/gui/page_ch_settings_adv.cpp +++ b/src/eez/apps/psu/gui/page_ch_settings_adv.cpp @@ -56,14 +56,13 @@ void ChSettingsAdvOptionsPage::toggleDprog() { //////////////////////////////////////////////////////////////////////////////// -void ChSettingsAdvRangesPage::onModeSet(uint8_t value) { +void ChSettingsAdvRangesPage::onModeSet(uint16_t value) { popPage(); g_channel->setCurrentRangeSelectionMode((CurrentRangeSelectionMode)value); } void ChSettingsAdvRangesPage::selectMode() { - pushSelectFromEnumPage(g_channelCurrentRangeSelectionModeEnumDefinition, - g_channel->getCurrentRangeSelectionMode(), 0, onModeSet); + pushSelectFromEnumPage(g_channelCurrentRangeSelectionModeEnumDefinition, g_channel->getCurrentRangeSelectionMode(), 0, onModeSet); } void ChSettingsAdvRangesPage::toggleAutoRanging() { @@ -78,36 +77,34 @@ void ChSettingsAdvViewPage::pageAlloc() { origYTViewRate = ytViewRate = g_channel->ytViewRate; } -bool ChSettingsAdvViewPage::isDisabledDisplayValue1(uint8_t value) { +bool ChSettingsAdvViewPage::isDisabledDisplayValue1(uint16_t value) { ChSettingsAdvViewPage *page = (ChSettingsAdvViewPage *)getPreviousPage(); return value == page->displayValue2; } -void ChSettingsAdvViewPage::onDisplayValue1Set(uint8_t value) { +void ChSettingsAdvViewPage::onDisplayValue1Set(uint16_t value) { popPage(); ChSettingsAdvViewPage *page = (ChSettingsAdvViewPage *)getActivePage(); - page->displayValue1 = value; + page->displayValue1 = (uint8_t)value; } void ChSettingsAdvViewPage::editDisplayValue1() { - pushSelectFromEnumPage(g_channelDisplayValueEnumDefinition, displayValue1, - isDisabledDisplayValue1, onDisplayValue1Set); + pushSelectFromEnumPage(g_channelDisplayValueEnumDefinition, displayValue1, isDisabledDisplayValue1, onDisplayValue1Set); } -bool ChSettingsAdvViewPage::isDisabledDisplayValue2(uint8_t value) { +bool ChSettingsAdvViewPage::isDisabledDisplayValue2(uint16_t value) { ChSettingsAdvViewPage *page = (ChSettingsAdvViewPage *)getPreviousPage(); return value == page->displayValue1; } -void ChSettingsAdvViewPage::onDisplayValue2Set(uint8_t value) { +void ChSettingsAdvViewPage::onDisplayValue2Set(uint16_t value) { popPage(); ChSettingsAdvViewPage *page = (ChSettingsAdvViewPage *)getActivePage(); - page->displayValue2 = value; + page->displayValue2 = (uint8_t)value; } void ChSettingsAdvViewPage::editDisplayValue2() { - pushSelectFromEnumPage(g_channelDisplayValueEnumDefinition, displayValue2, - isDisabledDisplayValue2, onDisplayValue2Set); + pushSelectFromEnumPage(g_channelDisplayValueEnumDefinition, displayValue2, isDisabledDisplayValue2, onDisplayValue2Set); } void ChSettingsAdvViewPage::onYTViewRateSet(float value) { diff --git a/src/eez/apps/psu/gui/page_ch_settings_adv.h b/src/eez/apps/psu/gui/page_ch_settings_adv.h index 8aca45901..555a0f66f 100644 --- a/src/eez/apps/psu/gui/page_ch_settings_adv.h +++ b/src/eez/apps/psu/gui/page_ch_settings_adv.h @@ -38,7 +38,7 @@ class ChSettingsAdvRangesPage : public Page { void toggleAutoRanging(); private: - static void onModeSet(uint8_t value); + static void onModeSet(uint16_t value); }; class ChSettingsAdvViewPage : public SetPage { @@ -62,10 +62,10 @@ class ChSettingsAdvViewPage : public SetPage { uint8_t origDisplayValue2; float origYTViewRate; - static bool isDisabledDisplayValue1(uint8_t value); - static void onDisplayValue1Set(uint8_t value); - static bool isDisabledDisplayValue2(uint8_t value); - static void onDisplayValue2Set(uint8_t value); + static bool isDisabledDisplayValue1(uint16_t value); + static void onDisplayValue1Set(uint16_t value); + static bool isDisabledDisplayValue2(uint16_t value); + static void onDisplayValue2Set(uint16_t value); static void onYTViewRateSet(float value); }; diff --git a/src/eez/apps/psu/gui/page_ch_settings_protection.cpp b/src/eez/apps/psu/gui/page_ch_settings_protection.cpp index 2b94c5070..9277df95e 100644 --- a/src/eez/apps/psu/gui/page_ch_settings_protection.cpp +++ b/src/eez/apps/psu/gui/page_ch_settings_protection.cpp @@ -174,9 +174,9 @@ void ChSettingsOvpProtectionPage::pageAlloc() { defLevel = channel_dispatcher::getUMax(*g_channel); origDelay = delay = MakeValue(g_channel->prot_conf.u_delay, UNIT_SECOND); - minDelay = g_channel->params->OVP_MIN_DELAY; - maxDelay = g_channel->params->OVP_MAX_DELAY; - defaultDelay = g_channel->params->OVP_DEFAULT_DELAY; + minDelay = g_channel->params.OVP_MIN_DELAY; + maxDelay = g_channel->params.OVP_MAX_DELAY; + defaultDelay = g_channel->params.OVP_DEFAULT_DELAY; } void ChSettingsOvpProtectionPage::onSetParamsOk() { @@ -209,9 +209,9 @@ void ChSettingsOcpProtectionPage::pageAlloc() { origLevel = level = 0; origDelay = delay = MakeValue(g_channel->prot_conf.i_delay, UNIT_SECOND); - minDelay = g_channel->params->OCP_MIN_DELAY; - maxDelay = g_channel->params->OCP_MAX_DELAY; - defaultDelay = g_channel->params->OCP_DEFAULT_DELAY; + minDelay = g_channel->params.OCP_MIN_DELAY; + maxDelay = g_channel->params.OCP_MAX_DELAY; + defaultDelay = g_channel->params.OCP_DEFAULT_DELAY; } void ChSettingsOcpProtectionPage::onSetParamsOk() { @@ -246,9 +246,9 @@ void ChSettingsOppProtectionPage::pageAlloc() { defLevel = channel_dispatcher::getOppDefaultLevel(*g_channel); origDelay = delay = MakeValue(g_channel->prot_conf.p_delay, UNIT_SECOND); - minDelay = g_channel->params->OPP_MIN_DELAY; - maxDelay = g_channel->params->OPP_MAX_DELAY; - defaultDelay = g_channel->params->OPP_DEFAULT_DELAY; + minDelay = g_channel->params.OPP_MIN_DELAY; + maxDelay = g_channel->params.OPP_MAX_DELAY; + defaultDelay = g_channel->params.OPP_DEFAULT_DELAY; } void ChSettingsOppProtectionPage::onSetParamsOk() { diff --git a/src/eez/apps/psu/gui/page_ch_settings_trigger.cpp b/src/eez/apps/psu/gui/page_ch_settings_trigger.cpp index 4d41f8bf3..d82014d65 100644 --- a/src/eez/apps/psu/gui/page_ch_settings_trigger.cpp +++ b/src/eez/apps/psu/gui/page_ch_settings_trigger.cpp @@ -51,12 +51,12 @@ void ChSettingsTriggerPage::onFinishTriggerModeSet() { profile::save(); } -void ChSettingsTriggerPage::onTriggerModeSet(uint8_t value) { +void ChSettingsTriggerPage::onTriggerModeSet(uint16_t value) { popPage(); if (channel_dispatcher::getVoltageTriggerMode(*g_channel) != value || channel_dispatcher::getCurrentTriggerMode(*g_channel) != value) { - g_newTriggerMode = value; + g_newTriggerMode = (uint8_t)value; if (trigger::isInitiated() || list::isActive()) { yesNoDialog(PAGE_ID_YES_NO, "Trigger is active. Are you sure?", onFinishTriggerModeSet, @@ -68,9 +68,7 @@ void ChSettingsTriggerPage::onTriggerModeSet(uint8_t value) { } void ChSettingsTriggerPage::editTriggerMode() { - pushSelectFromEnumPage(g_channelTriggerModeEnumDefinition, - channel_dispatcher::getVoltageTriggerMode(*g_channel), 0, - onTriggerModeSet); + pushSelectFromEnumPage(g_channelTriggerModeEnumDefinition, channel_dispatcher::getVoltageTriggerMode(*g_channel), 0, onTriggerModeSet); } void ChSettingsTriggerPage::onVoltageTriggerValueSet(float value) { @@ -127,16 +125,14 @@ void ChSettingsTriggerPage::toggleOutputState() { profile::save(); } -void ChSettingsTriggerPage::onTriggerOnListStopSet(uint8_t value) { +void ChSettingsTriggerPage::onTriggerOnListStopSet(uint16_t value) { popPage(); channel_dispatcher::setTriggerOnListStop(*g_channel, (TriggerOnListStop)value); profile::save(); } void ChSettingsTriggerPage::editTriggerOnListStop() { - pushSelectFromEnumPage(g_channelTriggerOnListStopEnumDefinition, - channel_dispatcher::getTriggerOnListStop(*g_channel), 0, - onTriggerOnListStopSet); + pushSelectFromEnumPage(g_channelTriggerOnListStopEnumDefinition, channel_dispatcher::getTriggerOnListStop(*g_channel), 0, onTriggerOnListStopSet); } void ChSettingsTriggerPage::onListCountSet(float value) { diff --git a/src/eez/apps/psu/gui/page_ch_settings_trigger.h b/src/eez/apps/psu/gui/page_ch_settings_trigger.h index fa766fb25..57d38cf6e 100644 --- a/src/eez/apps/psu/gui/page_ch_settings_trigger.h +++ b/src/eez/apps/psu/gui/page_ch_settings_trigger.h @@ -44,9 +44,9 @@ class ChSettingsTriggerPage : public Page { private: static void onFinishTriggerModeSet(); - static void onTriggerModeSet(uint8_t value); + static void onTriggerModeSet(uint16_t value); - static void onTriggerOnListStopSet(uint8_t value); + static void onTriggerOnListStopSet(uint16_t value); static void onVoltageTriggerValueSet(float value); static void onCurrentTriggerValueSet(float value); diff --git a/src/eez/apps/psu/gui/page_sys_settings.cpp b/src/eez/apps/psu/gui/page_sys_settings.cpp index 57c567310..bcd70f202 100644 --- a/src/eez/apps/psu/gui/page_sys_settings.cpp +++ b/src/eez/apps/psu/gui/page_sys_settings.cpp @@ -150,7 +150,7 @@ void SysSettingsDateTimePage::edit() { } } -void SysSettingsDateTimePage::onDstRuleSet(uint8_t value) { +void SysSettingsDateTimePage::onDstRuleSet(uint16_t value) { popPage(); SysSettingsDateTimePage *page = (SysSettingsDateTimePage *)getActivePage(); page->dstRule = (datetime::DstRule)value; @@ -641,7 +641,7 @@ void SysSettingsTriggerPage::pageAlloc() { trigger::isContinuousInitializationEnabled(); } -void SysSettingsTriggerPage::onTriggerSourceSet(uint8_t value) { +void SysSettingsTriggerPage::onTriggerSourceSet(uint16_t value) { popPage(); SysSettingsTriggerPage *page = (SysSettingsTriggerPage *)getActivePage(); page->m_source = (trigger::Source)value; @@ -717,7 +717,7 @@ void SysSettingsIOPinsPage::togglePolarity() { m_polarity[i] = m_polarity[i] == io_pins::POLARITY_NEGATIVE ? io_pins::POLARITY_POSITIVE : io_pins::POLARITY_NEGATIVE; } -void SysSettingsIOPinsPage::onFunctionSet(uint8_t value) { +void SysSettingsIOPinsPage::onFunctionSet(uint16_t value) { popPage(); SysSettingsIOPinsPage *page = (SysSettingsIOPinsPage *)getActivePage(); page->m_function[page->pinNumber] = (io_pins::Function)value; @@ -767,7 +767,7 @@ void SysSettingsSerialPage::toggle() { m_enabled = !m_enabled; } -void SysSettingsSerialPage::onParitySet(uint8_t value) { +void SysSettingsSerialPage::onParitySet(uint16_t value) { popPage(); SysSettingsSerialPage *page = (SysSettingsSerialPage *)getActivePage(); page->m_parity = (serial::Parity)value; diff --git a/src/eez/apps/psu/gui/page_sys_settings.h b/src/eez/apps/psu/gui/page_sys_settings.h index af6cf846b..709caded6 100644 --- a/src/eez/apps/psu/gui/page_sys_settings.h +++ b/src/eez/apps/psu/gui/page_sys_settings.h @@ -57,7 +57,7 @@ class SysSettingsDateTimePage : public SetPage { int16_t origTimeZone; datetime::DstRule origDstRule; - static void onDstRuleSet(uint8_t value); + static void onDstRuleSet(uint16_t value); #if OPTION_ETHERNET enum { @@ -256,7 +256,7 @@ class SysSettingsTriggerPage : public SetPage { float m_delayOrig; bool m_initiateContinuouslyOrig; - static void onTriggerSourceSet(uint8_t value); + static void onTriggerSourceSet(uint16_t value); static void onDelaySet(float value); }; @@ -279,7 +279,7 @@ class SysSettingsIOPinsPage : public SetPage { io_pins::Polarity m_polarityOrig[NUM_IO_PINS]; io_pins::Function m_functionOrig[NUM_IO_PINS]; - static void onFunctionSet(uint8_t value); + static void onFunctionSet(uint16_t value); }; class SysSettingsSerialPage : public SetPage { @@ -301,7 +301,7 @@ class SysSettingsSerialPage : public SetPage { uint8_t m_baudIndexOrig; serial::Parity m_parityOrig; - static void onParitySet(uint8_t value); + static void onParitySet(uint16_t value); }; class SysSettingsTrackingPage : public SetPage { diff --git a/src/eez/apps/psu/gui/psu.cpp b/src/eez/apps/psu/gui/psu.cpp index 888a3fb57..f70db0b73 100644 --- a/src/eez/apps/psu/gui/psu.cpp +++ b/src/eez/apps/psu/gui/psu.cpp @@ -596,9 +596,9 @@ void onSetPowerTripDelay(float delay) { void changePowerTripDelay(int iChannel) { g_iChannelSetValue = iChannel; Channel &channel = Channel::get(iChannel); - float minDelay = channel.params->OPP_MIN_DELAY; - float maxDelay = channel.params->OPP_MAX_DELAY; - float defaultDelay = channel.params->OPP_DEFAULT_DELAY; + float minDelay = channel.params.OPP_MIN_DELAY; + float maxDelay = channel.params.OPP_MAX_DELAY; + float defaultDelay = channel.params.OPP_DEFAULT_DELAY; changeValue(channel, MakeValue(channel.prot_conf.p_delay, UNIT_SECOND), minDelay, maxDelay, defaultDelay, onSetPowerTripDelay); diff --git a/src/eez/apps/psu/init.cpp b/src/eez/apps/psu/init.cpp index 4aa8d9e5e..b96f00ef1 100644 --- a/src/eez/apps/psu/init.cpp +++ b/src/eez/apps/psu/init.cpp @@ -102,7 +102,7 @@ void oneIter() { } bool measureAllAdcValuesOnChannel(int channelIndex) { - if (g_slots[Channel::get(channelIndex).slotIndex].moduleType == MODULE_TYPE_NONE) { + if (g_slots[Channel::get(channelIndex).slotIndex].moduleInfo->moduleType == MODULE_TYPE_NONE) { return true; } diff --git a/src/eez/apps/psu/profile.cpp b/src/eez/apps/psu/profile.cpp index 2a624a603..57e71ba4d 100644 --- a/src/eez/apps/psu/profile.cpp +++ b/src/eez/apps/psu/profile.cpp @@ -190,7 +190,7 @@ void recallChannelsFromProfile(Parameters *profile, int location) { channel.flags.outputEnabled = channel.isTripped() ? 0 : profile->channels[i].flags.output_enabled; channel.flags.senseEnabled = profile->channels[i].flags.sense_enabled; - if (channel.getFeatures() & CH_FEATURE_RPROG) { + if (channel.params.features & CH_FEATURE_RPROG) { channel.flags.rprogEnabled = profile->channels[i].flags.rprog_enabled; } else { channel.flags.rprogEnabled = 0; @@ -244,14 +244,15 @@ void fillProfile(Parameters *pProfile) { for (int i = 0; i < CH_MAX; ++i) { Channel &channel = Channel::get(i); if (i < CH_NUM && channel.isInstalled()) { - profile.channels[i].flags.moduleType = g_slots[channel.slotIndex].moduleType; + profile.channels[i].moduleType = g_slots[channel.slotIndex].moduleInfo->moduleType; + profile.channels[i].moduleRevision = g_slots[channel.slotIndex].moduleRevision; profile.channels[i].flags.parameters_are_valid = 1; profile.channels[i].flags.output_enabled = channel.flags.outputEnabled; profile.channels[i].flags.sense_enabled = channel.flags.senseEnabled; - if (channel.getFeatures() & CH_FEATURE_RPROG) { + if (channel.params.features & CH_FEATURE_RPROG) { profile.channels[i].flags.rprog_enabled = channel.flags.rprogEnabled; } else { @@ -381,7 +382,7 @@ void tick() { bool checkProfileModuleMatch(Parameters *profile) { for (int i = 0; i < CH_NUM; ++i) { Channel &channel = Channel::get(i); - if (profile->channels[i].flags.parameters_are_valid && profile->channels[i].flags.moduleType != g_slots[channel.slotIndex].moduleType) { + if (profile->channels[i].flags.parameters_are_valid && profile->channels[i].moduleType != g_slots[channel.slotIndex].moduleInfo->moduleType) { return false; // mismatch } } diff --git a/src/eez/apps/psu/profile.h b/src/eez/apps/psu/profile.h index 888daaf78..c496cc0d3 100644 --- a/src/eez/apps/psu/profile.h +++ b/src/eez/apps/psu/profile.h @@ -29,7 +29,6 @@ namespace profile { /// Channel binary flags stored in profile. struct ChannelFlags { - unsigned moduleType: 8; unsigned output_enabled : 1; unsigned sense_enabled : 1; unsigned u_state : 1; @@ -52,6 +51,8 @@ struct ChannelFlags { /// Channel parameters stored in profile. struct ChannelParameters { + uint16_t moduleType; + uint16_t moduleRevision; ChannelFlags flags; float u_set; float u_step; @@ -93,7 +94,7 @@ struct Parameters { temperature::ProtectionConfiguration temp_prot[temp_sensor::MAX_NUM_TEMP_SENSORS]; }; -static const uint16_t PROFILE_VERSION = 10; +static const uint16_t PROFILE_VERSION = 11; // auto save support extern bool g_profileDirty; diff --git a/src/eez/apps/psu/psu.cpp b/src/eez/apps/psu/psu.cpp index bfbbef227..3af7995d0 100644 --- a/src/eez/apps/psu/psu.cpp +++ b/src/eez/apps/psu/psu.cpp @@ -223,34 +223,27 @@ void init() { int channelIndex = 0; for (uint8_t slotIndex = 0; slotIndex < NUM_SLOTS; slotIndex++) { - uint16_t value; - if (!bp3c::eeprom::read(slotIndex, (uint8_t *)&value, 2, (uint16_t)0)) { - g_slots[slotIndex].moduleType = MODULE_TYPE_NONE; - g_slots[slotIndex].boardRevision = CH_BOARD_REVISION_NONE; - } else if (value == 405) { - g_slots[slotIndex].moduleType = MODULE_TYPE_DCP405; - g_slots[slotIndex].boardRevision = CH_BOARD_REVISION_DCP405_R1B1; - } else if (value == 406) { - g_slots[slotIndex].moduleType = MODULE_TYPE_DCP405; - g_slots[slotIndex].boardRevision = CH_BOARD_REVISION_DCP405_R2B5; - } else if (value == 505) { - g_slots[slotIndex].moduleType = MODULE_TYPE_DCP505; - g_slots[slotIndex].boardRevision = CH_BOARD_REVISION_DCP505_R1B3; - } else if (value == 220) { - g_slots[slotIndex].moduleType = MODULE_TYPE_DCM220; - g_slots[slotIndex].boardRevision = CH_BOARD_REVISION_DCM220_R1B1; - } else { - g_slots[slotIndex].moduleType = MODULE_TYPE_NONE; - g_slots[slotIndex].boardRevision = CH_BOARD_REVISION_NONE; + auto& slot = g_slots[slotIndex]; + + static const uint16_t ADDRESS = 0; + uint16_t value[2]; + if (!bp3c::eeprom::read(slotIndex, (uint8_t *)&value, sizeof(value), ADDRESS)) { + value[0] = MODULE_TYPE_NONE; + value[1] = 0; } - g_slots[slotIndex].channelIndex = channelIndex; + uint16_t moduleType = value[0]; + uint16_t moduleRevision = value[1]; + + slot.moduleInfo = getModuleInfo(moduleType); + slot.moduleRevision = moduleRevision; + slot.channelIndex = channelIndex; - for (uint8_t subchannelIndex = 0; subchannelIndex < g_modules[g_slots[slotIndex].moduleType].numChannels; subchannelIndex++) { - Channel::get(channelIndex++).set(slotIndex, subchannelIndex, g_slots[slotIndex].boardRevision); + for (uint8_t subchannelIndex = 0; subchannelIndex < slot.moduleInfo->numChannels; subchannelIndex++) { + Channel::get(channelIndex++).set(slotIndex, subchannelIndex); } - if (g_slots[slotIndex].moduleType != MODULE_TYPE_NONE) { + if (slot.moduleInfo->moduleType != MODULE_TYPE_NONE) { persist_conf::loadModuleConf(slotIndex); ontime::g_moduleCounters[slotIndex].init(); } @@ -616,7 +609,7 @@ bool powerUp() { ontime::g_mcuCounter.start(); for (int slotIndex = 0; slotIndex < NUM_SLOTS; slotIndex++) { - if (g_slots[slotIndex].moduleType != MODULE_TYPE_NONE) { + if (g_slots[slotIndex].moduleInfo->moduleType != MODULE_TYPE_NONE) { ontime::g_moduleCounters[slotIndex].start(); } } @@ -684,7 +677,7 @@ void powerDown() { ontime::g_mcuCounter.stop(); for (int slotIndex = 0; slotIndex < NUM_SLOTS; slotIndex++) { - if (g_slots[slotIndex].moduleType != MODULE_TYPE_NONE) { + if (g_slots[slotIndex].moduleInfo->moduleType != MODULE_TYPE_NONE) { ontime::g_moduleCounters[slotIndex].stop(); } } @@ -824,7 +817,7 @@ void onProtectionTripped() { void tick_onTimeCounters(uint32_t tickCount) { ontime::g_mcuCounter.tick(tickCount); for (int slotIndex = 0; slotIndex < NUM_SLOTS; slotIndex++) { - if (g_slots[slotIndex].moduleType != MODULE_TYPE_NONE) { + if (g_slots[slotIndex].moduleInfo->moduleType != MODULE_TYPE_NONE) { ontime::g_moduleCounters[slotIndex].tick(tickCount); } } diff --git a/src/eez/apps/psu/psu.h b/src/eez/apps/psu/psu.h index 30d395a25..633edb1a8 100644 --- a/src/eez/apps/psu/psu.h +++ b/src/eez/apps/psu/psu.h @@ -26,7 +26,6 @@ #include #include -#include #include #include diff --git a/src/eez/apps/psu/scpi/debug.cpp b/src/eez/apps/psu/scpi/debug.cpp index 3592b3559..ce9f042bd 100644 --- a/src/eez/apps/psu/scpi/debug.cpp +++ b/src/eez/apps/psu/scpi/debug.cpp @@ -32,10 +32,6 @@ #include #include -#if defined(EEZ_PLATFORM_STM32) -#include -#endif - extern "C" { #include "py/compile.h" #include "py/runtime.h" @@ -459,307 +455,11 @@ scpi_result_t scpi_cmd_debugPythonQ(scpi_t *context) { #endif // DEBUG } -#if defined(EEZ_PLATFORM_STM32) -#define BUFFER_SIZE 10 - -static uint8_t g_output[BUFFER_SIZE]; -static uint8_t g_input[BUFFER_SIZE]; - -static bool g_synchronized; - -#define SPI_SLAVE_SYNBYTE 0x53 -#define SPI_MASTER_SYNBYTE 0xAC - -void specDelayUs(uint32_t microseconds) { - while (microseconds--) { - // 216 NOP's - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - __ASM volatile ("NOP"); - } -} - -void masterSynchro(void) { - uint8_t txackbytes = SPI_MASTER_SYNBYTE, rxackbytes = 0x00; - do { - taskENTER_CRITICAL(); - spi::selectA(2); - spi::transfer(2, &txackbytes, &rxackbytes, 1); - spi::deselectA(2); - taskEXIT_CRITICAL(); - } while(rxackbytes != SPI_SLAVE_SYNBYTE); - - while (HAL_GPIO_ReadPin(SPI5_IRQ_GPIO_Port, SPI5_IRQ_Pin) != GPIO_PIN_SET); -} -#endif - scpi_result_t scpi_cmd_debugDcm220Q(scpi_t *context) { #if defined(EEZ_PLATFORM_STM32) - int32_t cmd; - if (!SCPI_ParamInt(context, &cmd, TRUE)) { - return SCPI_RES_ERR; - } - - int32_t param; - int mandatoryParam = cmd == 14 || cmd == 15 || cmd == 24 || cmd == 25; - if (!SCPI_ParamInt(context, ¶m, mandatoryParam)) { - if (mandatoryParam || SCPI_ParamErrorOccurred(context)) { - return SCPI_RES_ERR; - } - } - - g_output[0] = (uint8_t)(cmd | 0x40); - g_output[1] = (uint8_t)(param & 0xFF); - g_output[2] = (uint8_t)((param >> 8) & 0xFF); - - for (int i = 3; i < BUFFER_SIZE; i++) { - g_output[i] = 0xFF; - } - - if (!g_synchronized) { - spi::init(2, spi::CHIP_DCM220); - masterSynchro(); - g_synchronized = true; - } - - if (cmd == 31) { - HAL_GPIO_WritePin(OE_SYNC_GPIO_Port, OE_SYNC_Pin, GPIO_PIN_RESET); - } - - taskENTER_CRITICAL(); - spi::selectA(2); - spi::transfer(2, g_output, g_input, BUFFER_SIZE); - spi::deselectA(2); - taskEXIT_CRITICAL(); - - if (cmd == 31) { - HAL_GPIO_WritePin(OE_SYNC_GPIO_Port, OE_SYNC_Pin, GPIO_PIN_SET); - } - char text[100]; - - sprintf(text, "0x%02X, 0x%02X, U_MON_1=%d, I_MON_1=%d, U_MON_2=%d, I_MON_2=%d", - (int)g_input[0], (int)g_input[1], - (int)(uint16_t)(g_input[2] | (g_input[3] << 8)), - (int)(uint16_t)(g_input[4] | (g_input[5] << 8)), - (int)(uint16_t)(g_input[6] | (g_input[7] << 8)), - (int)(uint16_t)(g_input[8] | (g_input[9] << 8))); - + sprintf(text, "TODO"); SCPI_ResultText(context, text); - return SCPI_RES_OK; #else SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); diff --git a/src/eez/apps/psu/scpi/diag.cpp b/src/eez/apps/psu/scpi/diag.cpp index 9ed00f363..0f4056a6e 100644 --- a/src/eez/apps/psu/scpi/diag.cpp +++ b/src/eez/apps/psu/scpi/diag.cpp @@ -430,7 +430,7 @@ scpi_result_t scpi_cmd_diagnosticInformationRegsQ(scpi_t *context) { for (int i = 0; i < CH_MAX; i++) { Channel& channel = Channel::get(i); if (channel.isInstalled()) { - if (g_slots[channel.slotIndex].moduleType == MODULE_TYPE_DCP405) { + if (g_slots[channel.slotIndex].moduleInfo->moduleType == MODULE_TYPE_DCP405 || g_slots[channel.slotIndex].moduleInfo->moduleType == MODULE_TYPE_DCP405B) { sprintf(buffer + strlen(buffer), "CH%d:\n", i + 1); sprintf(buffer + strlen(buffer), "\tIOEXP:\n"); diff --git a/src/eez/apps/psu/scpi/outp.cpp b/src/eez/apps/psu/scpi/outp.cpp index 690217844..43950612b 100644 --- a/src/eez/apps/psu/scpi/outp.cpp +++ b/src/eez/apps/psu/scpi/outp.cpp @@ -215,7 +215,7 @@ scpi_result_t scpi_cmd_outputDprog(scpi_t *context) { return SCPI_RES_ERR; } - if (channel->getFeatures() & CH_FEATURE_DPROG) { + if (channel->params.features & CH_FEATURE_DPROG) { channel->setDprogState((DprogState)dprogState); } else { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); @@ -231,7 +231,7 @@ scpi_result_t scpi_cmd_outputDprogQ(scpi_t *context) { return SCPI_RES_ERR; } - if (channel->getFeatures() & CH_FEATURE_DPROG) { + if (channel->params.features & CH_FEATURE_DPROG) { resultChoiceName(context, dprogStateChoice, channel->getDprogState()); } else { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); diff --git a/src/eez/apps/psu/scpi/simu.cpp b/src/eez/apps/psu/scpi/simu.cpp index 4b0c5ae39..a36af3827 100644 --- a/src/eez/apps/psu/scpi/simu.cpp +++ b/src/eez/apps/psu/scpi/simu.cpp @@ -179,7 +179,7 @@ scpi_result_t scpi_cmd_simulatorVoltageProgramExternal(scpi_t *context) { return SCPI_RES_ERR; } - if (!(channel->getFeatures() & CH_FEATURE_RPROG)) { + if (!(channel->params.features & CH_FEATURE_RPROG)) { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); return SCPI_RES_ERR; } @@ -201,7 +201,7 @@ scpi_result_t scpi_cmd_simulatorVoltageProgramExternalQ(scpi_t *context) { return SCPI_RES_ERR; } - if (!(channel->getFeatures() & CH_FEATURE_RPROG)) { + if (!(channel->params.features & CH_FEATURE_RPROG)) { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); return SCPI_RES_ERR; } @@ -256,7 +256,7 @@ scpi_result_t scpi_cmd_simulatorRpol(scpi_t *context) { return SCPI_RES_ERR; } - if (channel->getFeatures() & CH_FEATURE_RPOL) { + if (channel->params.features & CH_FEATURE_RPOL) { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); return SCPI_RES_ERR; } @@ -278,7 +278,7 @@ scpi_result_t scpi_cmd_simulatorRpolQ(scpi_t *context) { return SCPI_RES_ERR; } - if (channel->getFeatures() & CH_FEATURE_RPOL) { + if (channel->params.features & CH_FEATURE_RPOL) { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); return SCPI_RES_ERR; } diff --git a/src/eez/apps/psu/scpi/sour.cpp b/src/eez/apps/psu/scpi/sour.cpp index d376a4619..7e5039747 100644 --- a/src/eez/apps/psu/scpi/sour.cpp +++ b/src/eez/apps/psu/scpi/sour.cpp @@ -254,7 +254,7 @@ scpi_result_t scpi_cmd_sourceCurrentLevelImmediateStepIncrement(scpi_t *context) return SCPI_RES_ERR; } - return set_step(context, &channel->i, channel->params->I_MIN_STEP, channel->params->I_MAX_STEP, channel->params->I_DEF_STEP, SCPI_UNIT_AMPER); + return set_step(context, &channel->i, channel->params.I_MIN_STEP, channel->params.I_MAX_STEP, channel->params.I_DEF_STEP, SCPI_UNIT_AMPER); } scpi_result_t scpi_cmd_sourceCurrentLevelImmediateStepIncrementQ(scpi_t *context) { @@ -264,7 +264,7 @@ scpi_result_t scpi_cmd_sourceCurrentLevelImmediateStepIncrementQ(scpi_t *context return SCPI_RES_ERR; } - return get_source_value(context, *channel, UNIT_AMPER, channel->i.step, channel->params->I_DEF_STEP); + return get_source_value(context, *channel, UNIT_AMPER, channel->i.step, channel->params.I_DEF_STEP); } scpi_result_t scpi_cmd_sourceVoltageLevelImmediateStepIncrement(scpi_t *context) { @@ -274,7 +274,7 @@ scpi_result_t scpi_cmd_sourceVoltageLevelImmediateStepIncrement(scpi_t *context) return SCPI_RES_ERR; } - return set_step(context, &channel->u, channel->params->U_MIN_STEP, channel->params->U_MAX_STEP, channel->params->U_DEF_STEP, SCPI_UNIT_VOLT); + return set_step(context, &channel->u, channel->params.U_MIN_STEP, channel->params.U_MAX_STEP, channel->params.U_DEF_STEP, SCPI_UNIT_VOLT); } scpi_result_t scpi_cmd_sourceVoltageLevelImmediateStepIncrementQ(scpi_t *context) { @@ -284,7 +284,7 @@ scpi_result_t scpi_cmd_sourceVoltageLevelImmediateStepIncrementQ(scpi_t *context return SCPI_RES_ERR; } - return get_source_value(context, *channel, UNIT_VOLT, channel->u.step, channel->params->U_DEF_STEP); + return get_source_value(context, *channel, UNIT_VOLT, channel->u.step, channel->params.U_DEF_STEP); } //////////////////////////////////////////////////////////////////////////////// @@ -297,7 +297,7 @@ scpi_result_t scpi_cmd_sourceCurrentProtectionDelayTime(scpi_t *context) { } float delay; - if (!get_duration_param(context, delay, channel->params->OCP_MIN_DELAY, channel->params->OCP_MAX_DELAY, channel->params->OCP_DEFAULT_DELAY)) { + if (!get_duration_param(context, delay, channel->params.OCP_MIN_DELAY, channel->params.OCP_MAX_DELAY, channel->params.OCP_DEFAULT_DELAY)) { return SCPI_RES_ERR; } @@ -396,7 +396,7 @@ scpi_result_t scpi_cmd_sourcePowerProtectionDelayTime(scpi_t *context) { } float delay; - if (!get_duration_param(context, delay, channel->params->OPP_MIN_DELAY, channel->params->OPP_MAX_DELAY, channel->params->OPP_DEFAULT_DELAY)) { + if (!get_duration_param(context, delay, channel->params.OPP_MIN_DELAY, channel->params.OPP_MAX_DELAY, channel->params.OPP_DEFAULT_DELAY)) { return SCPI_RES_ERR; } @@ -495,7 +495,7 @@ scpi_result_t scpi_cmd_sourceVoltageProtectionDelayTime(scpi_t *context) { } float delay; - if (!get_duration_param(context, delay, channel->params->OVP_MIN_DELAY, channel->params->OVP_MAX_DELAY, channel->params->OVP_DEFAULT_DELAY)) { + if (!get_duration_param(context, delay, channel->params.OVP_MIN_DELAY, channel->params.OVP_MAX_DELAY, channel->params.OVP_DEFAULT_DELAY)) { return SCPI_RES_ERR; } @@ -565,7 +565,7 @@ scpi_result_t scpi_cmd_sourceVoltageProtectionType(scpi_t *context) { return SCPI_RES_ERR; } - if (g_slots[channel->slotIndex].moduleType != MODULE_TYPE_DCP405) { + if (!(channel->params.features & CH_FEATURE_HW_OVP)) { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); return SCPI_RES_ERR; } @@ -586,7 +586,7 @@ scpi_result_t scpi_cmd_sourceVoltageProtectionTypeQ(scpi_t *context) { return SCPI_RES_ERR; } - if (g_slots[channel->slotIndex].moduleType != MODULE_TYPE_DCP405) { + if (!(channel->params.features & CH_FEATURE_HW_OVP)) { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); return SCPI_RES_ERR; } @@ -651,7 +651,7 @@ scpi_result_t scpi_cmd_sourceVoltageProgramSource(scpi_t *context) { return SCPI_RES_ERR; } - if (!(channel->getFeatures() & CH_FEATURE_RPROG)) { + if (!(channel->params.features & CH_FEATURE_RPROG)) { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); return SCPI_RES_ERR; } @@ -683,7 +683,7 @@ scpi_result_t scpi_cmd_sourceVoltageProgramSourceQ(scpi_t *context) { return SCPI_RES_ERR; } - if (!(channel->getFeatures() & CH_FEATURE_RPROG)) { + if (!(channel->params.features & CH_FEATURE_RPROG)) { SCPI_ErrorPush(context, SCPI_ERROR_HARDWARE_MISSING); return SCPI_RES_ERR; } diff --git a/src/eez/apps/psu/scpi/syst.cpp b/src/eez/apps/psu/scpi/syst.cpp index b17ea33af..3b43a1012 100644 --- a/src/eez/apps/psu/scpi/syst.cpp +++ b/src/eez/apps/psu/scpi/syst.cpp @@ -461,7 +461,7 @@ scpi_result_t scpi_cmd_systemChannelInformationPowerQ(scpi_t *context) { return SCPI_RES_ERR; } - SCPI_ResultFloat(context, channel->params->PTOT); + SCPI_ResultFloat(context, channel->params.PTOT); return SCPI_RES_OK; } @@ -473,7 +473,7 @@ scpi_result_t scpi_cmd_systemChannelInformationProgramQ(scpi_t *context) { return SCPI_RES_ERR; } - uint16_t features = channel->getFeatures(); + uint16_t features = channel->params.features; char strFeatures[64] = { 0 }; @@ -565,13 +565,19 @@ scpi_result_t scpi_cmd_systemChannelInformationOntimeLastQ(scpi_t *context) { } scpi_result_t scpi_cmd_systemChannelModelQ(scpi_t *context) { - // TODO migrate to generic firmware Channel *channel = param_channel(context, false, true); if (!channel) { return SCPI_RES_ERR; } - SCPI_ResultText(context, channel->getBoardAndRevisionName()); + if (channel->isInstalled()) { + auto &slot = g_slots[channel->slotIndex]; + char text[100]; + sprintf(text, "%s_R%dB%d", slot.moduleInfo->moduleName, (int)(slot.moduleRevision >> 8), (int)(slot.moduleRevision & 0xFF)); + SCPI_ResultText(context, text); + } else { + SCPI_ResultText(context, "None"); + } return SCPI_RES_OK; } diff --git a/src/eez/apps/psu/temp_sensor.cpp b/src/eez/apps/psu/temp_sensor.cpp index 51748eb31..a7444dfe8 100644 --- a/src/eez/apps/psu/temp_sensor.cpp +++ b/src/eez/apps/psu/temp_sensor.cpp @@ -62,10 +62,7 @@ bool TempSensor::isInstalled() { } else if (type >= CH1 && type <= CH6) { int channelIndex = type - CH1; int slotIndex = Channel::get(channelIndex).slotIndex; - return - g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405 || - g_slots[slotIndex].moduleType == MODULE_TYPE_DCP505 || - g_slots[slotIndex].moduleType == MODULE_TYPE_DCM220; + return g_slots[slotIndex].moduleInfo->moduleType != MODULE_TYPE_NONE; } else { return false; } @@ -100,16 +97,17 @@ float TempSensor::doRead() { if (type >= CH1 && type <= CH6) { int channelIndex = type - CH1; int slotIndex = Channel::get(channelIndex).slotIndex; + auto &slot = g_slots[slotIndex]; - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405 && Channel::get(slotIndex).boardRevision == CH_BOARD_REVISION_DCP405_R2B5) { + if ((slot.moduleInfo->moduleType == MODULE_TYPE_DCP405 && slot.moduleRevision >= MODULE_REVISION_DCP405_R1B1) || slot.moduleInfo->moduleType == MODULE_TYPE_DCP405B) { return drivers::tc77::readTemperature(slotIndex); } - if ((g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405 && Channel::get(slotIndex).boardRevision == CH_BOARD_REVISION_DCP405_R1B1) || g_slots[slotIndex].moduleType == MODULE_TYPE_DCP505) { + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP405 || slot.moduleInfo->moduleType == MODULE_TYPE_DCP505) { return drivers::tmp1075::readTemperature(slotIndex); } - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCM220) { + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCM220) { return dcm220::readTemperature(channelIndex); } } diff --git a/src/eez/gui/action_impl.cpp b/src/eez/gui/action_impl.cpp index 5c0c5b9da..3cf4d4aed 100644 --- a/src/eez/gui/action_impl.cpp +++ b/src/eez/gui/action_impl.cpp @@ -934,9 +934,9 @@ void themesEnumDefinition(data::DataOperationEnum operation, data::Cursor &curso } -void onSetSelectedThemeIndex(uint8_t value) { +void onSetSelectedThemeIndex(uint16_t value) { popPage(); - persist_conf::devConf2.selectedThemeIndex = value; + persist_conf::devConf2.selectedThemeIndex = (uint8_t)value; persist_conf::saveDevice2(); mcu::display::onThemeChanged(); refreshScreen(); @@ -1021,7 +1021,7 @@ void action_user_switch_clicked() { static int g_slotIndex; -void onSetModuleType(uint8_t moduleType) { +void onSetModuleType(uint16_t moduleType) { popPage(); bp3c::eeprom::writeModuleType(g_slotIndex, moduleType); @@ -1035,7 +1035,7 @@ void onSetModuleType(uint8_t moduleType) { void selectSlot(int slotIndex) { g_slotIndex = slotIndex; - pushSelectFromEnumPage(g_moduleTypeEnumDefinition, g_slots[slotIndex].moduleType, NULL, onSetModuleType); + pushSelectFromEnumPage(g_moduleTypeEnumDefinition, g_slots[slotIndex].moduleInfo->moduleType, NULL, onSetModuleType); } void action_front_panel_select_slot1() { diff --git a/src/eez/gui/app_context.cpp b/src/eez/gui/app_context.cpp index fabb67305..c28b56caf 100644 --- a/src/eez/gui/app_context.cpp +++ b/src/eez/gui/app_context.cpp @@ -326,15 +326,15 @@ void AppContext::showPageOnNextIter(int pageId) { m_pageIdToSetOnNextIter = pageId; } -void AppContext::pushSelectFromEnumPage(const data::EnumItem *enumDefinition, uint8_t currentValue, - bool (*disabledCallback)(uint8_t value), - void (*onSet)(uint8_t)) { +void AppContext::pushSelectFromEnumPage(const data::EnumItem *enumDefinition, uint16_t currentValue, + bool (*disabledCallback)(uint16_t value), + void (*onSet)(uint16_t)) { m_selectFromEnumPage.init(enumDefinition, currentValue, disabledCallback, onSet); pushPage(INTERNAL_PAGE_ID_SELECT_FROM_ENUM, &m_selectFromEnumPage); } void AppContext::pushSelectFromEnumPage(void (*enumDefinitionFunc)(data::DataOperationEnum operation, data::Cursor &cursor, data::Value &value), - uint8_t currentValue, bool (*disabledCallback)(uint8_t value), void (*onSet)(uint8_t)) { + uint16_t currentValue, bool (*disabledCallback)(uint16_t value), void (*onSet)(uint16_t)) { m_selectFromEnumPage.init(enumDefinitionFunc, currentValue, disabledCallback, onSet); pushPage(INTERNAL_PAGE_ID_SELECT_FROM_ENUM, &m_selectFromEnumPage); } diff --git a/src/eez/gui/app_context.h b/src/eez/gui/app_context.h index 4b28c73c6..265e3f11d 100644 --- a/src/eez/gui/app_context.h +++ b/src/eez/gui/app_context.h @@ -82,10 +82,10 @@ class AppContext { int getPreviousPageId(); Page *getPreviousPage(); - void pushSelectFromEnumPage(const data::EnumItem *enumDefinition, uint8_t currentValue, - bool (*disabledCallback)(uint8_t value), void (*onSet)(uint8_t)); + void pushSelectFromEnumPage(const data::EnumItem *enumDefinition, uint16_t currentValue, + bool (*disabledCallback)(uint16_t value), void (*onSet)(uint16_t)); void pushSelectFromEnumPage(void (*enumDefinitionFunc)(data::DataOperationEnum operation, data::Cursor &cursor, data::Value &value), - uint8_t currentValue, bool (*disabledCallback)(uint8_t value), void (*onSet)(uint8_t)); + uint16_t currentValue, bool (*disabledCallback)(uint16_t value), void (*onSet)(uint16_t)); void replacePage(int pageId, Page *page = nullptr); diff --git a/src/eez/gui/data.cpp b/src/eez/gui/data.cpp index cbf1fb059..00b9e1c6c 100644 --- a/src/eez/gui/data.cpp +++ b/src/eez/gui/data.cpp @@ -194,15 +194,32 @@ bool compare_SLOT_INFO_value(const Value &a, const Value &b) { void SLOT_INFO_value_to_text(const Value &value, char *text, int count) { int slotIndex = value.getInt(); - psu::Channel &channel = psu::Channel::get(slotIndex); + auto &slot = g_slots[slotIndex]; + psu::Channel &channel = psu::Channel::get(slot.channelIndex); if (channel.isInstalled()) { - snprintf(text, count - 1, "%s %s", channel.getBoardName(), channel.getRevisionName()); + snprintf(text, count - 1, "%s R%dB%d", slot.moduleInfo->moduleName, (int)(slot.moduleRevision >> 8), (int)(slot.moduleRevision & 0xFF)); } else { strncpy(text, "Not installed", count - 1); } text[count - 1] = 0; } +bool compare_SLOT_INFO2_value(const Value &a, const Value &b) { + return a.getInt() == b.getInt(); +} + +void SLOT_INFO2_value_to_text(const Value &value, char *text, int count) { + int slotIndex = value.getInt(); + auto &slot = g_slots[slotIndex]; + psu::Channel &channel = psu::Channel::get(slot.channelIndex); + if (channel.isInstalled()) { + snprintf(text, count - 1, "%s_R%dB%d", slot.moduleInfo->moduleName, (int)(slot.moduleRevision >> 8), (int)(slot.moduleRevision & 0xFF)); + } else { + strncpy(text, "None", count - 1); + } + text[count - 1] = 0; +} + bool compare_TEST_RESULT_value(const Value &a, const Value &b) { return a.getInt() == b.getInt(); } @@ -232,7 +249,7 @@ static CompareValueFunction g_compareBuiltInValueFunctions[] = { compare_NONE_value, compare_INT_value, compare_FLOAT_value, compare_STR_value, compare_ENUM_value, compare_SCPI_ERROR_value, compare_PERCENTAGE_value, compare_SIZE_value, compare_POINTER_value, - compare_PAGE_INFO_value, compare_MASTER_INFO_value, compare_SLOT_INFO_value, + compare_PAGE_INFO_value, compare_MASTER_INFO_value, compare_SLOT_INFO_value, compare_SLOT_INFO2_value, compare_TEST_RESULT_value }; @@ -240,7 +257,7 @@ static ValueToTextFunction g_builtInValueToTextFunctions[] = { NONE_value_to_text, INT_value_to_text, FLOAT_value_to_text, STR_value_to_text, ENUM_value_to_text, SCPI_ERROR_value_to_text, PERCENTAGE_value_to_text, SIZE_value_to_text, POINTER_value_to_text, - PAGE_INFO_value_to_text, MASTER_INFO_value_to_text, SLOT_INFO_value_to_text, + PAGE_INFO_value_to_text, MASTER_INFO_value_to_text, SLOT_INFO_value_to_text, SLOT_INFO2_value_to_text, TEST_RESULT_value_to_text }; @@ -516,14 +533,6 @@ void data_slots(DataOperationEnum operation, Cursor &cursor, Value &value) { } } -void data_slot_module_type(DataOperationEnum operation, Cursor &cursor, Value &value) { - if (operation == data::DATA_OPERATION_GET) { - int slotIndex = cursor.i; - assert(slotIndex >= 0 && slotIndex < CH_MAX); - value = g_slots[slotIndex].moduleType; - } -} - #endif void data_selected_theme(DataOperationEnum operation, Cursor &cursor, Value &value) { diff --git a/src/eez/gui/data.h b/src/eez/gui/data.h index 5b2702209..11220bd67 100644 --- a/src/eez/gui/data.h +++ b/src/eez/gui/data.h @@ -37,6 +37,7 @@ enum BuiltInValueType { VALUE_TYPE_PAGE_INFO, VALUE_TYPE_MASTER_INFO, VALUE_TYPE_SLOT_INFO, + VALUE_TYPE_SLOT_INFO2, VALUE_TYPE_TEST_RESULT, VALUE_TYPE_USER, }; @@ -50,7 +51,7 @@ struct Style; namespace data { struct EnumItem { - uint8_t value; + uint16_t value; const char *menuLabel; const char *widgetLabel; }; @@ -58,8 +59,8 @@ struct EnumItem { extern const data::EnumItem *g_enumDefinitions[]; struct EnumValue { - uint8_t enumValue; - uint8_t enumDefinition; + uint16_t enumValue; + uint16_t enumDefinition; }; struct PairOfUint8Value { diff --git a/src/eez/gui/gui.cpp b/src/eez/gui/gui.cpp index e2f672d0f..fed068c5e 100644 --- a/src/eez/gui/gui.cpp +++ b/src/eez/gui/gui.cpp @@ -73,7 +73,6 @@ bool onSystemStateChanged() { if (eez::g_systemState == eez::SystemState::BOOTING) { if (eez::g_systemStatePhase == 0) { g_guiTaskHandle = osThreadCreate(osThread(g_guiTask), nullptr); - } else if (eez::g_systemStatePhase == 1) { } } @@ -182,13 +181,13 @@ bool isPageActiveOrOnStack(int pageId) { return g_appContext->isPageActiveOrOnStack(pageId); } -void pushSelectFromEnumPage(const data::EnumItem *enumDefinition, uint8_t currentValue, - bool (*disabledCallback)(uint8_t value), void (*onSet)(uint8_t)) { +void pushSelectFromEnumPage(const data::EnumItem *enumDefinition, uint16_t currentValue, + bool (*disabledCallback)(uint16_t value), void (*onSet)(uint16_t)) { g_appContext->pushSelectFromEnumPage(enumDefinition, currentValue, disabledCallback, onSet); } void pushSelectFromEnumPage(void(*enumDefinitionFunc)(data::DataOperationEnum operation, data::Cursor &cursor, data::Value &value), - uint8_t currentValue, bool(*disabledCallback)(uint8_t value), void(*onSet)(uint8_t)) { + uint16_t currentValue, bool(*disabledCallback)(uint16_t value), void(*onSet)(uint16_t)) { g_appContext->pushSelectFromEnumPage(enumDefinitionFunc, currentValue, disabledCallback, onSet); } @@ -640,4 +639,4 @@ void animateRects(Buffer startBuffer, int numRects, float duration) { } // namespace gui } // namespace eez -#endif \ No newline at end of file +#endif diff --git a/src/eez/gui/gui.h b/src/eez/gui/gui.h index 537f6ad57..c410adabb 100644 --- a/src/eez/gui/gui.h +++ b/src/eez/gui/gui.h @@ -60,10 +60,10 @@ int getPreviousPageId(); Page *getPreviousPage(); Page *getPage(int pageId); bool isPageActiveOrOnStack(int pageId); -void pushSelectFromEnumPage(const data::EnumItem *enumDefinition, uint8_t currentValue, - bool (*disabledCallback)(uint8_t value), void (*onSet)(uint8_t)); +void pushSelectFromEnumPage(const data::EnumItem *enumDefinition, uint16_t currentValue, + bool (*disabledCallback)(uint16_t value), void (*onSet)(uint16_t)); void pushSelectFromEnumPage(void(*enumDefinitionFunc)(data::DataOperationEnum operation, data::Cursor &cursor, data::Value &value), - uint8_t currentValue, bool(*disabledCallback)(uint8_t value), void(*onSet)(uint8_t)); + uint16_t currentValue, bool(*disabledCallback)(uint16_t value), void(*onSet)(uint16_t)); bool isPageInternal(int pageId); void executeAction(int actionId); diff --git a/src/eez/gui/page.cpp b/src/eez/gui/page.cpp index 63c29ebc1..273a360be 100644 --- a/src/eez/gui/page.cpp +++ b/src/eez/gui/page.cpp @@ -316,8 +316,8 @@ void ToastMessagePage::executeAction() { //////////////////////////////////////////////////////////////////////////////// -void SelectFromEnumPage::init(const data::EnumItem *enumDefinition_, uint8_t currentValue_, - bool (*disabledCallback_)(uint8_t value), void (*onSet_)(uint8_t)) +void SelectFromEnumPage::init(const data::EnumItem *enumDefinition_, uint16_t currentValue_, + bool (*disabledCallback_)(uint16_t value), void (*onSet_)(uint16_t)) { enumDefinition = enumDefinition_; enumDefinitionFunc = NULL; @@ -329,7 +329,7 @@ void SelectFromEnumPage::init(const data::EnumItem *enumDefinition_, uint8_t cur } void SelectFromEnumPage::init(void (*enumDefinitionFunc_)(data::DataOperationEnum operation, data::Cursor &cursor, data::Value &value), - uint8_t currentValue_, bool (*disabledCallback_)(uint8_t value), void (*onSet_)(uint8_t)) + uint16_t currentValue_, bool (*disabledCallback_)(uint16_t value), void (*onSet_)(uint16_t)) { enumDefinition = NULL; enumDefinitionFunc = enumDefinitionFunc_; @@ -380,7 +380,7 @@ void SelectFromEnumPage::init() { findPagePosition(); } -uint8_t SelectFromEnumPage::getValue(int i) { +uint16_t SelectFromEnumPage::getValue(int i) { if (enumDefinitionFunc) { data::Value value; data::Cursor cursor(i); diff --git a/src/eez/gui/page.h b/src/eez/gui/page.h index c7f33fefa..0bcf28986 100644 --- a/src/eez/gui/page.h +++ b/src/eez/gui/page.h @@ -110,10 +110,10 @@ class ToastMessagePage : public InternalPage { class SelectFromEnumPage : public InternalPage { public: - void init(const data::EnumItem *enumDefinition_, uint8_t currentValue_, - bool (*disabledCallback_)(uint8_t value), void (*onSet_)(uint8_t)); + void init(const data::EnumItem *enumDefinition_, uint16_t currentValue_, + bool (*disabledCallback_)(uint16_t value), void (*onSet_)(uint16_t)); void init(void (*enumDefinitionFunc)(data::DataOperationEnum operation, data::Cursor &cursor, data::Value &value), - uint8_t currentValue_, bool (*disabledCallback_)(uint8_t value), void (*onSet_)(uint8_t)); + uint16_t currentValue_, bool (*disabledCallback_)(uint16_t value), void (*onSet_)(uint16_t)); void init(); @@ -133,12 +133,12 @@ class SelectFromEnumPage : public InternalPage { bool dirty; - uint8_t getValue(int i); + uint16_t getValue(int i); const char *getLabel(int i); - uint8_t currentValue; - bool (*disabledCallback)(uint8_t value); - void (*onSet)(uint8_t); + uint16_t currentValue; + bool (*disabledCallback)(uint16_t value); + void (*onSet)(uint16_t); void findPagePosition(); diff --git a/src/eez/index.cpp b/src/eez/index.cpp index 5811bdd49..ed2b607c5 100644 --- a/src/eez/index.cpp +++ b/src/eez/index.cpp @@ -129,45 +129,64 @@ OnSystemStateChangedCallback g_onSystemStateChangedCallbacks[] = { psu::onSystemStateChanged, }; -int g_numOnSystemStateChangedCallbacks = - sizeof(g_onSystemStateChangedCallbacks) / sizeof(OnSystemStateChangedCallback); +int g_numOnSystemStateChangedCallbacks = sizeof(g_onSystemStateChangedCallbacks) / sizeof(OnSystemStateChangedCallback); -ModuleInfo g_modules[] = { +static ModuleInfo g_modules[] = { { + MODULE_TYPE_NONE, + "None", 0, - CH_BOARD_REVISION_NONE, 1, nullptr }, { - 406, - CH_BOARD_REVISION_DCP405_R2B5, + MODULE_TYPE_DCP405, + "DCP405", + MODULE_REVISION_DCP405_R2B7, 1, dcpX05::g_channelInterfaces }, { - 220, - CH_BOARD_REVISION_DCM220_R1B1, - 2, - dcm220::g_channelInterfaces + MODULE_TYPE_DCP405B, + "DCP405B", + MODULE_REVISION_DCP405B_R2B7, + 1, + dcpX05::g_channelInterfaces }, { - 505, - CH_BOARD_REVISION_DCP505_R1B3, + MODULE_TYPE_DCP505, + "DCP505", + MODULE_REVISION_DCP505_R1B3, 1, dcpX05::g_channelInterfaces }, + { + MODULE_TYPE_DCM220, + "DCM220", + MODULE_REVISION_DCM220_R2B4, + 2, + dcm220::g_channelInterfaces + } }; +ModuleInfo *getModuleInfo(uint16_t moduleType) { + for (size_t i = 0; i < sizeof(g_modules) / sizeof(ModuleInfo); i++) { + if (g_modules[i].moduleType == moduleType) { + return &g_modules[i]; + } + } + return &g_modules[0]; +} + SlotInfo g_slots[NUM_SLOTS] = { { - MODULE_TYPE_NONE + &g_modules[0] }, { - MODULE_TYPE_NONE + &g_modules[0] }, { - MODULE_TYPE_NONE + &g_modules[0] } }; diff --git a/src/eez/index.h b/src/eez/index.h index 7f2b8029b..cf43d658f 100644 --- a/src/eez/index.h +++ b/src/eez/index.h @@ -53,11 +53,95 @@ enum DprogState { DPROG_STATE_AUTO = 2 }; +enum ChannelFeatures { + CH_FEATURE_VOLT = (1 << 0), + CH_FEATURE_CURRENT = (1 << 1), + CH_FEATURE_POWER = (1 << 2), + CH_FEATURE_OE = (1 << 3), + CH_FEATURE_DPROG = (1 << 4), + CH_FEATURE_RPROG = (1 << 5), + CH_FEATURE_RPOL = (1 << 6), + CH_FEATURE_CURRENT_DUAL_RANGE = (1 << 7), + CH_FEATURE_HW_OVP = (1 << 8), + CH_FEATURE_COUPLING = (1 << 9) +}; + +struct ChannelParams { + float U_MIN; + float U_DEF; + float U_MAX; + float U_MAX_CONF; + + float U_MIN_STEP; + float U_DEF_STEP; + float U_MAX_STEP; + + float U_CAL_VAL_MIN; + float U_CAL_VAL_MID; + float U_CAL_VAL_MAX; + float U_CURR_CAL; + + float I_MIN; + float I_DEF; + float I_MAX; + + float I_MAX_CONF; + float I_MIN_STEP; + float I_DEF_STEP; + float I_MAX_STEP; + + float I_CAL_VAL_MIN; + float I_CAL_VAL_MID; + float I_CAL_VAL_MAX; + float I_VOLT_CAL; + + bool OVP_DEFAULT_STATE; + float OVP_MIN_DELAY; + float OVP_DEFAULT_DELAY; + float OVP_MAX_DELAY; + + bool OCP_DEFAULT_STATE; + float OCP_MIN_DELAY; + float OCP_DEFAULT_DELAY; + float OCP_MAX_DELAY; + + bool OPP_DEFAULT_STATE; + float OPP_MIN_DELAY; + float OPP_DEFAULT_DELAY; + float OPP_MAX_DELAY; + float OPP_MIN_LEVEL; + float OPP_DEFAULT_LEVEL; + float OPP_MAX_LEVEL; + + float PTOT; + + float U_RESOLUTION; + float I_RESOLUTION; + float I_LOW_RESOLUTION; + float P_RESOLUTION; + + float VOLTAGE_GND_OFFSET; // [V], (1375 / 65535) * (40V | 50V) + float CURRENT_GND_OFFSET; // [A] + + /// Maximum difference, in percentage, between ADC + /// and real value during calibration. + float CALIBRATION_DATA_TOLERANCE_PERCENT; + + /// Maximum difference, in percentage, between calculated mid value + /// and real mid value during calibration. + float CALIBRATION_MID_TOLERANCE_PERCENT; + + /// Returns features present (check ChannelFeatures) in board revision of this channel. + uint32_t features; +}; + struct ChannelInterface { int slotIndex; ChannelInterface(int slotIndex); + virtual void getParams(int subchannelIndex, ChannelParams ¶ms) = 0; + virtual void init(int subchannelIndex) = 0; virtual void reset(int subchannelIndex) = 0; virtual void test(int subchannelIndex) = 0; @@ -108,23 +192,35 @@ struct ChannelInterface { #endif }; -static const uint8_t MODULE_TYPE_NONE = 0; -static const uint8_t MODULE_TYPE_DCP405 = 1; -static const uint8_t MODULE_TYPE_DCM220 = 2; -static const uint8_t MODULE_TYPE_DCP505 = 3; +static const uint16_t MODULE_TYPE_NONE = 0; +static const uint16_t MODULE_TYPE_DCP405 = 405; +static const uint16_t MODULE_TYPE_DCP405B = 406; +static const uint16_t MODULE_TYPE_DCP505 = 505; +static const uint16_t MODULE_TYPE_DCM220 = 220; + +static const uint16_t MODULE_REVISION_DCP405_R1B1 = 0x0101; +static const uint16_t MODULE_REVISION_DCP405_R2B5 = 0x0205; +static const uint16_t MODULE_REVISION_DCP405_R2B7 = 0x0207; + +static const uint16_t MODULE_REVISION_DCP405B_R2B7 = 0x0207; + +static const uint16_t MODULE_REVISION_DCP505_R1B3 = 0x0103; + +static const uint16_t MODULE_REVISION_DCM220_R2B4 = 0x0204; struct ModuleInfo { - uint16_t moduleId; - uint8_t lasestBoardRevision; // TODO should be lasestModuleRevision + uint16_t moduleType; + const char *moduleName; + uint16_t latestModuleRevision; uint8_t numChannels; ChannelInterface **channelInterfaces; }; -extern ModuleInfo g_modules[]; +ModuleInfo *getModuleInfo(uint16_t moduleType); struct SlotInfo { - uint8_t moduleType; // MODULE_TYPE_... - uint8_t boardRevision; // TODO should be moduleRevision; + ModuleInfo *moduleInfo; + uint16_t moduleRevision; uint8_t channelIndex; }; diff --git a/src/eez/modules/aux_ps/fan.cpp b/src/eez/modules/aux_ps/fan.cpp index f3d1c69d0..0f2746196 100644 --- a/src/eez/modules/aux_ps/fan.cpp +++ b/src/eez/modules/aux_ps/fan.cpp @@ -328,8 +328,8 @@ void getIMonMax(float &iMonMax, float &iMax) { iMonMax = channel.i.mon; } } - if (channel.params->I_MAX > iMax) { - iMax = channel.params->I_MAX; + if (channel.params.I_MAX > iMax) { + iMax = channel.params.I_MAX; } } } diff --git a/src/eez/modules/bp3c/eeprom.cpp b/src/eez/modules/bp3c/eeprom.cpp index 5ad273c5d..bd2139315 100644 --- a/src/eez/modules/bp3c/eeprom.cpp +++ b/src/eez/modules/bp3c/eeprom.cpp @@ -223,12 +223,12 @@ bool test() { return g_testResult != TEST_FAILED; } -void writeModuleType(uint8_t slotIndex, uint8_t moduleType) { - uint8_t buffer[] = { - (uint8_t)(g_modules[moduleType].moduleId & 0xFF), - (uint8_t)((g_modules[moduleType].moduleId >> 8) & 0xFF), +void writeModuleType(uint8_t slotIndex, uint16_t moduleType) { + uint16_t buffer[] = { + moduleType, + getModuleInfo(moduleType)->latestModuleRevision }; - write(slotIndex, buffer, 2, 0); + write(slotIndex, (uint8_t *)buffer, 4, 0); } } // namespace eeprom diff --git a/src/eez/modules/bp3c/eeprom.h b/src/eez/modules/bp3c/eeprom.h index 6671b558e..8d806aa9a 100644 --- a/src/eez/modules/bp3c/eeprom.h +++ b/src/eez/modules/bp3c/eeprom.h @@ -107,7 +107,7 @@ extern TestResult g_testResult; bool read(uint8_t slotIndex, uint8_t *buffer, uint16_t buffer_size, uint16_t address); bool write(uint8_t slotIndex, const uint8_t *buffer, uint16_t buffer_size, uint16_t address); -void writeModuleType(uint8_t slotIndex, uint8_t moduleType); +void writeModuleType(uint8_t slotIndex, uint16_t moduleType); } // namespace eeprom } // namespace bp3c diff --git a/src/eez/modules/dcm220/channel.cpp b/src/eez/modules/dcm220/channel.cpp index 7f98603cc..1cad38c7f 100644 --- a/src/eez/modules/dcm220/channel.cpp +++ b/src/eez/modules/dcm220/channel.cpp @@ -148,6 +148,70 @@ struct Channel : ChannelInterface { #endif } + void getParams(int subchannelIndex, ChannelParams ¶ms) { + params.U_MIN = 0.0f; + params.U_DEF = 0.0f; + params.U_MAX = 20.0f; + params.U_MAX_CONF = 20.0f; + + params.U_MIN_STEP = 0.01f; + params.U_DEF_STEP = 0.1f; + params.U_MAX_STEP = 5.0f; + + params.U_CAL_VAL_MIN = 0.75f; + params.U_CAL_VAL_MID = 10.0f; + params.U_CAL_VAL_MAX = 18.0f; + params.U_CURR_CAL = 20.0f; + + params.I_MIN = 0.0f; + params.I_DEF = 0.0f; + params.I_MAX = 4.0f; + params.I_MAX_CONF = 4.0f; + + params.I_MIN_STEP = 0.01f; + params.I_DEF_STEP = 0.01f; + params.I_MAX_STEP = 1.0f; + + params.I_CAL_VAL_MIN = 0.05f; + params.I_CAL_VAL_MID = 1.95f; + params.I_CAL_VAL_MAX = 3.8f; + params.I_VOLT_CAL = 0.5f; + + params.OVP_DEFAULT_STATE = false; + params.OVP_MIN_DELAY = 0.0f; + params.OVP_DEFAULT_DELAY = 0.005f; + params.OVP_MAX_DELAY = 10.0f; + + params.OCP_DEFAULT_STATE = false; + params.OCP_MIN_DELAY = 0.0f; + params.OCP_DEFAULT_DELAY = 0.02f; + params.OCP_MAX_DELAY = 10.0f; + + params.OPP_DEFAULT_STATE = true; + params.OPP_MIN_DELAY = 1.0f; + params.OPP_DEFAULT_DELAY = 10.0f; + params.OPP_MAX_DELAY = 300.0f; + params.OPP_MIN_LEVEL = 0.0f; + params.OPP_DEFAULT_LEVEL = 80.0f; + params.OPP_MAX_LEVEL = 80.0f; + + params.PTOT = 80.0f; + + params.U_RESOLUTION = 0.01f; + params.I_RESOLUTION = 0.02f; + params.I_LOW_RESOLUTION = 0; + params.P_RESOLUTION = 0.001f; + + params.VOLTAGE_GND_OFFSET = 0; + params.CURRENT_GND_OFFSET = 0; + + params.CALIBRATION_DATA_TOLERANCE_PERCENT = 15.0f; + + params.CALIBRATION_MID_TOLERANCE_PERCENT = 2.0f; + + params.features = CH_FEATURE_VOLT | CH_FEATURE_CURRENT | CH_FEATURE_POWER | CH_FEATURE_OE; + } + #if defined(EEZ_PLATFORM_STM32) bool isCrcOk; @@ -297,11 +361,11 @@ struct Channel : ChannelInterface { int offset = subchannelIndex * 2; uint16_t uMonAdc = inputSetValues[offset]; - float uMon = remap(uMonAdc, (float)ADC_MIN, channel.params->U_MIN, (float)ADC_MAX, channel.params->U_MAX); + float uMon = remap(uMonAdc, (float)ADC_MIN, channel.params.U_MIN, (float)ADC_MAX, channel.params.U_MAX); channel.onAdcData(ADC_DATA_TYPE_U_MON, uMon); uint16_t iMonAdc = inputSetValues[offset + 1]; - float iMon = remap(iMonAdc, (float)ADC_MIN, channel.params->I_MIN, (float)ADC_MAX, channel.params->I_MAX); + float iMon = remap(iMonAdc, (float)ADC_MIN, channel.params.I_MIN, (float)ADC_MAX, channel.params.I_MAX); channel.onAdcData(ADC_DATA_TYPE_I_MON, iMon); } #endif @@ -397,14 +461,14 @@ struct Channel : ChannelInterface { #if defined(EEZ_PLATFORM_SIMULATOR) psu::Channel &channel = psu::Channel::getBySlotIndex(slotIndex, subchannelIndex); - uSet[subchannelIndex] = remap(clamp((float)value, (float)DAC_MIN, (float)DAC_MAX), (float)DAC_MIN, channel.params->U_MIN, (float)DAC_MAX, channel.params->U_MAX); + uSet[subchannelIndex] = remap(clamp((float)value, (float)DAC_MIN, (float)DAC_MAX), (float)DAC_MIN, channel.params.U_MIN, (float)DAC_MAX, channel.params.U_MAX); #endif } void setDacVoltageFloat(int subchannelIndex, float value) { #if defined(EEZ_PLATFORM_STM32) psu::Channel &channel = psu::Channel::getBySlotIndex(slotIndex, subchannelIndex); - value = remap(value, channel.params->U_MIN, (float)DAC_MIN, channel.params->U_MAX, (float)DAC_MAX); + value = remap(value, channel.params.U_MIN, (float)DAC_MIN, channel.params.U_MAX, (float)DAC_MAX); uSet[subchannelIndex] = (uint16_t)clamp(round(value), DAC_MIN, DAC_MAX); #endif @@ -421,14 +485,14 @@ struct Channel : ChannelInterface { #if defined(EEZ_PLATFORM_SIMULATOR) psu::Channel &channel = psu::Channel::getBySlotIndex(slotIndex, subchannelIndex); - uSet[subchannelIndex] = remap(clamp((float)value, (float)DAC_MIN, (float)DAC_MAX), (float)DAC_MIN, channel.params->I_MIN, (float)DAC_MAX, channel.params->I_MAX); + uSet[subchannelIndex] = remap(clamp((float)value, (float)DAC_MIN, (float)DAC_MAX), (float)DAC_MIN, channel.params.I_MIN, (float)DAC_MAX, channel.params.I_MAX); #endif } void setDacCurrentFloat(int subchannelIndex, float value) { #if defined(EEZ_PLATFORM_STM32) psu::Channel &channel = psu::Channel::getBySlotIndex(slotIndex, subchannelIndex); - value = remap(value, channel.params->I_MIN, (float)DAC_MIN, channel.params->I_MAX, (float)DAC_MAX); + value = remap(value, channel.params.I_MIN, (float)DAC_MIN, channel.params.I_MAX, (float)DAC_MAX); iSet[subchannelIndex] = (uint16_t)clamp(round(value), DAC_MIN, DAC_MAX); #endif diff --git a/src/eez/modules/dcpX05/adc.cpp b/src/eez/modules/dcpX05/adc.cpp index 46867f2e3..2641cc652 100644 --- a/src/eez/modules/dcpX05/adc.cpp +++ b/src/eez/modules/dcpX05/adc.cpp @@ -34,17 +34,17 @@ namespace psu { #if defined(EEZ_PLATFORM_STM32) float remapAdcDataToVoltage(Channel& channel, AdcDataType adcDataType, int16_t adcData) { - float value = remap((float)adcData, (float)AnalogDigitalConverter::ADC_MIN, channel.params->U_MIN, (float)AnalogDigitalConverter::ADC_MAX, channel.params->U_MAX_CONF); + float value = remap((float)adcData, (float)AnalogDigitalConverter::ADC_MIN, channel.params.U_MIN, (float)AnalogDigitalConverter::ADC_MAX, channel.params.U_MAX_CONF); #if !defined(EEZ_PLATFORM_SIMULATOR) if (adcDataType == ADC_DATA_TYPE_U_MON || !channel.flags.rprogEnabled) { - value -= channel.params->VOLTAGE_GND_OFFSET; + value -= channel.params.VOLTAGE_GND_OFFSET; } #endif return value; } float remapAdcDataToCurrent(Channel& channel, AdcDataType adcDataType, int16_t adcData) { - float value = remap((float)adcData, (float)AnalogDigitalConverter::ADC_MIN, channel.params->I_MIN, (float)AnalogDigitalConverter::ADC_MAX, channel.getDualRangeMax()); + float value = remap((float)adcData, (float)AnalogDigitalConverter::ADC_MIN, channel.params.I_MIN, (float)AnalogDigitalConverter::ADC_MAX, channel.getDualRangeMax()); #if !defined(EEZ_PLATFORM_SIMULATOR) value -= channel.getDualRangeGndOffset(); #endif diff --git a/src/eez/modules/dcpX05/channel.cpp b/src/eez/modules/dcpX05/channel.cpp index 91998e6ac..601e31e22 100644 --- a/src/eez/modules/dcpX05/channel.cpp +++ b/src/eez/modules/dcpX05/channel.cpp @@ -64,6 +64,108 @@ struct Channel : ChannelInterface { Channel(int slotIndex_) : ChannelInterface(slotIndex_), uSet(0) {} + void getParams(int subchannelIndex, ChannelParams ¶ms) { + auto slot = g_slots[slotIndex]; + + params.U_MIN = 0.0f; + params.U_DEF = 0.0f; + + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP505) { + params.U_MAX = 50.0f; + params.U_MAX_CONF = 50.0f; + + params.U_CAL_VAL_MIN = 0.15f; + params.U_CAL_VAL_MID = 24.1f; + params.U_CAL_VAL_MAX = 48.0f; + params.U_CURR_CAL = 25.0f; + } else { + params.U_MAX = 40.0f; + params.U_MAX_CONF = 40.0f; + + params.U_CAL_VAL_MIN = 0.15f; + params.U_CAL_VAL_MID = 20.0f; + params.U_CAL_VAL_MAX = 38.0f; + params.U_CURR_CAL = 20.0f; + } + + params.U_MIN_STEP = 0.01f; + params.U_DEF_STEP = 0.1f; + params.U_MAX_STEP = 5.0f; + + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP505) { + params.U_CAL_VAL_MIN = 0.15f; + params.U_CAL_VAL_MID = 24.1f; + params.U_CAL_VAL_MAX = 48.0f; + params.U_CURR_CAL = 25.0f; + } else { + params.U_CAL_VAL_MIN = 0.15f; + params.U_CAL_VAL_MID = 20.0f; + params.U_CAL_VAL_MAX = 38.0f; + params.U_CURR_CAL = 20.0f; + } + + params.I_MIN = 0.0f; + params.I_DEF = 0.0f; + params.I_MAX = 5.0f; + params.I_MAX_CONF = 5.0f; + params.I_MIN_STEP = 0.01f; + params.I_DEF_STEP = 0.01f; + params.I_MAX_STEP = 1.0f; + params.I_CAL_VAL_MIN = 0.05f; + params.I_CAL_VAL_MID = 2.425f; + params.I_CAL_VAL_MAX = 4.8f; + params.I_VOLT_CAL = 0.1f; + + params.OVP_DEFAULT_STATE = false; + params.OVP_MIN_DELAY = 0.0f; + params.OVP_DEFAULT_DELAY = 0.005f; + params.OVP_MAX_DELAY = 10.0f; + + params.OCP_DEFAULT_STATE = false; + params.OCP_MIN_DELAY = 0.0f; + params.OCP_DEFAULT_DELAY = 0.02f; + params.OCP_MAX_DELAY = 10.0f; + + params.OPP_DEFAULT_STATE = true; + params.OPP_MIN_DELAY = 1.0f; + params.OPP_DEFAULT_DELAY = 10.0f; + params.OPP_MAX_DELAY = 300.0f; + params.OPP_MIN_LEVEL = 0.0f; + params.OPP_DEFAULT_LEVEL = 155.0f; + params.OPP_MAX_LEVEL = 160.0f; + + params.PTOT = 155.0f; + + params.U_RESOLUTION = 0.005f; + params.I_RESOLUTION = 0.0005f; + params.I_LOW_RESOLUTION = 0.000005f; + params.P_RESOLUTION = 0.001f; + + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP505) { + params.VOLTAGE_GND_OFFSET = 1.05f; + } else { + params.VOLTAGE_GND_OFFSET = 0.86f; + } + params.CURRENT_GND_OFFSET = 0.11f; + + params.CALIBRATION_DATA_TOLERANCE_PERCENT = 10.0f; + + params.CALIBRATION_MID_TOLERANCE_PERCENT = 1.0f; + + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP505) { + params.features = CH_FEATURE_VOLT | CH_FEATURE_CURRENT | CH_FEATURE_POWER | CH_FEATURE_OE | + CH_FEATURE_DPROG | CH_FEATURE_RPROG | CH_FEATURE_RPOL; + } else if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP405) { + params.features = CH_FEATURE_VOLT | CH_FEATURE_CURRENT | CH_FEATURE_POWER | CH_FEATURE_OE | + CH_FEATURE_DPROG | CH_FEATURE_RPROG | CH_FEATURE_RPOL | + CH_FEATURE_CURRENT_DUAL_RANGE | CH_FEATURE_HW_OVP | CH_FEATURE_COUPLING; + } else { + // DCP405B + params.features = CH_FEATURE_VOLT | CH_FEATURE_CURRENT | CH_FEATURE_POWER | CH_FEATURE_OE | + CH_FEATURE_CURRENT_DUAL_RANGE | CH_FEATURE_COUPLING; + } + } + void init(int subchannelIndex) { ioexp.slotIndex = slotIndex; adc.slotIndex = slotIndex; @@ -138,45 +240,47 @@ struct Channel : ChannelInterface { #endif } - if (dprogState == DPROG_STATE_AUTO) { - // turn off DP after delay - if (delayed_dp_off && (micros() - delayed_dp_off_start) >= DP_OFF_DELAY_PERIOD * 1000000L) { - delayed_dp_off = false; - setDpEnable(false); - } - } - - /// Output power is monitored and if its go below DP_NEG_LEV - /// that is negative value in Watts (default -1 W), - /// and that condition lasts more then DP_NEG_DELAY seconds (default 5 s), - /// down-programmer circuit has to be switched off. - if (channel.isOutputEnabled()) { - if (channel.u.mon_last * channel.i.mon_last >= DP_NEG_LEV || tickCount < dpNegMonitoringTime) { - dpNegMonitoringTime = tickCount; - } else { - if (tickCount - dpNegMonitoringTime > DP_NEG_DELAY * 1000000UL) { - if (dpOn) { - // DebugTrace("CH%d, neg. P, DP off: %f", channel.channelIndex + 1, channel.u.mon_last * channel.i.mon_last); - dpNegMonitoringTime = tickCount; - generateError(SCPI_ERROR_CH1_DOWN_PROGRAMMER_SWITCHED_OFF + channel.channelIndex); - setDpEnable(false); - } else { - // DebugTrace("CH%d, neg. P, output off: %f", channel.channelIndex + 1, channel.u.mon_last * channel.i.mon_last); - generateError(SCPI_ERROR_CH1_OUTPUT_FAULT_DETECTED - channel.channelIndex); - channel_dispatcher::outputEnable(channel, false); + if (channel.params.features & CH_FEATURE_DPROG) { + if (dprogState == DPROG_STATE_AUTO) { + // turn off DP after delay + if (delayed_dp_off && (micros() - delayed_dp_off_start) >= DP_OFF_DELAY_PERIOD * 1000000L) { + delayed_dp_off = false; + setDpEnable(false); + } + } + + /// Output power is monitored and if its go below DP_NEG_LEV + /// that is negative value in Watts (default -1 W), + /// and that condition lasts more then DP_NEG_DELAY seconds (default 5 s), + /// down-programmer circuit has to be switched off. + if (channel.isOutputEnabled()) { + if (channel.u.mon_last * channel.i.mon_last >= DP_NEG_LEV || tickCount < dpNegMonitoringTime) { + dpNegMonitoringTime = tickCount; + } else { + if (tickCount - dpNegMonitoringTime > DP_NEG_DELAY * 1000000UL) { + if (dpOn) { + // DebugTrace("CH%d, neg. P, DP off: %f", channel.channelIndex + 1, channel.u.mon_last * channel.i.mon_last); + dpNegMonitoringTime = tickCount; + generateError(SCPI_ERROR_CH1_DOWN_PROGRAMMER_SWITCHED_OFF + channel.channelIndex); + setDpEnable(false); + } else { + // DebugTrace("CH%d, neg. P, output off: %f", channel.channelIndex + 1, channel.u.mon_last * channel.i.mon_last); + generateError(SCPI_ERROR_CH1_OUTPUT_FAULT_DETECTED - channel.channelIndex); + channel_dispatcher::outputEnable(channel, false); + } + } else if (tickCount - dpNegMonitoringTime > 500 * 1000UL) { + if (dpOn && channel.channelIndex < 2) { + if (channel_dispatcher::getCouplingType() == channel_dispatcher::COUPLING_TYPE_SERIES) { + psu::Channel &channel2 = psu::Channel::get(channel.channelIndex == 0 ? 1 : 0); + voltageBalancing(channel2); + dpNegMonitoringTime = tickCount; + } else if (channel_dispatcher::getCouplingType() == channel_dispatcher::COUPLING_TYPE_PARALLEL) { + psu::Channel &channel2 = psu::Channel::get(channel.channelIndex == 0 ? 1 : 0); + currentBalancing(channel2); + dpNegMonitoringTime = tickCount; + } + } } - } else if (tickCount - dpNegMonitoringTime > 500 * 1000UL) { - if (dpOn && channel.channelIndex < 2) { - if (channel_dispatcher::getCouplingType() == channel_dispatcher::COUPLING_TYPE_SERIES) { - psu::Channel &channel2 = psu::Channel::get(channel.channelIndex == 0 ? 1 : 0); - voltageBalancing(channel2); - dpNegMonitoringTime = tickCount; - } else if (channel_dispatcher::getCouplingType() == channel_dispatcher::COUPLING_TYPE_PARALLEL) { - psu::Channel &channel2 = psu::Channel::get(channel.channelIndex == 0 ? 1 : 0); - currentBalancing(channel2); - dpNegMonitoringTime = tickCount; - } - } } } } @@ -195,11 +299,12 @@ struct Channel : ChannelInterface { } } - if (fallingEdge) { - fallingEdge = channel.u.mon_last > channel.u.set * (1.0f + CONF_OVP_PERCENTAGE / 100.0f); - } + if (channel.params.features & CH_FEATURE_HW_OVP) { + if (fallingEdge) { + fallingEdge = channel.u.mon_last > channel.u.set * (1.0f + CONF_OVP_PERCENTAGE / 100.0f); + } - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { + // HW OVP handling if (channel.isOutputEnabled()) { if (!fallingEdge && channel.isHwOvpEnabled() && !ioexp.testBit(IOExpander::DCP405_IO_BIT_OUT_OVP_ENABLE)) { if (isSwOvpAtStart) { @@ -300,14 +405,17 @@ struct Channel : ChannelInterface { } void setDpEnable(bool enable) { - // DP bit is active low - ioexp.changeBit(IOExpander::IO_BIT_OUT_DP_ENABLE, !enable); + psu::Channel &channel = psu::Channel::getBySlotIndex(slotIndex); + if (channel.params.features & CH_FEATURE_DPROG) { + // DP bit is active low + ioexp.changeBit(IOExpander::IO_BIT_OUT_DP_ENABLE, !enable); - setOperBits(OPER_ISUM_DP_OFF, !enable); - dpOn = enable; + setOperBits(OPER_ISUM_DP_OFF, !enable); + dpOn = enable; - if (enable) { - dpNegMonitoringTime = micros(); + if (enable) { + dpNegMonitoringTime = micros(); + } } } @@ -315,12 +423,14 @@ struct Channel : ChannelInterface { psu::Channel &channel = psu::Channel::getBySlotIndex(slotIndex); if (enable) { - if (dprogState == DPROG_STATE_AUTO) { - // enable DP - dpNegMonitoringTime = micros(); - delayed_dp_off = false; - setDpEnable(true); - } + if (channel.params.features & CH_FEATURE_DPROG) { + if (dprogState == DPROG_STATE_AUTO) { + // enable DP + dpNegMonitoringTime = micros(); + delayed_dp_off = false; + setDpEnable(true); + } + } dac.setVoltage(uSet); @@ -351,14 +461,16 @@ struct Channel : ChannelInterface { setCurrentRange(subchannelIndex); - if (dprogState == DPROG_STATE_AUTO) { - // turn off DP after some delay - delayed_dp_off = true; - delayed_dp_off_start = micros(); - } + if (channel.params.features & CH_FEATURE_DPROG) { + if (dprogState == DPROG_STATE_AUTO) { + // turn off DP after some delay + delayed_dp_off = true; + delayed_dp_off_start = micros(); + } + } } - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { + if (channel.params.features & CH_FEATURE_COUPLING) { if (channel.channelIndex == 0 && channel_dispatcher::getCouplingType() == channel_dispatcher::COUPLING_TYPE_PARALLEL) { ioexp.changeBit(IOExpander::DCP405_IO_BIT_OUT_OE_UNCOUPLED_LED, false); ioexp.changeBit(IOExpander::DCP405_IO_BIT_OUT_OE_COUPLED_LED, enable); @@ -395,18 +507,20 @@ struct Channel : ChannelInterface { } void setDprogState(DprogState dprogState_) { - dprogState = dprogState_; - - if (dprogState == DPROG_STATE_OFF) { - setDpEnable(false); - } else if (dprogState == DPROG_STATE_ON) { - setDpEnable(true); - } else { - psu::Channel &channel = psu::Channel::getBySlotIndex(slotIndex); - setDpEnable(channel.isOutputEnabled()); - } + psu::Channel &channel = psu::Channel::getBySlotIndex(slotIndex); + if (channel.params.features & CH_FEATURE_DPROG) { + dprogState = dprogState_; + + if (dprogState == DPROG_STATE_OFF) { + setDpEnable(false); + } else if (dprogState == DPROG_STATE_ON) { + setDpEnable(true); + } else { + setDpEnable(channel.isOutputEnabled()); + } - delayed_dp_off = false; + delayed_dp_off = false; + } } void setRemoteSense(int subchannelIndex, bool enable) { @@ -465,13 +579,14 @@ struct Channel : ChannelInterface { } void setCurrentRange(int subchannelIndex) { - psu::Channel &channel = psu::Channel::getBySlotIndex(slotIndex); + auto &channel = psu::Channel::getBySlotIndex(slotIndex); + auto &slot = g_slots[slotIndex]; if (!channel.hasSupportForCurrentDualRange()) { return; } - if (channel.boardRevision == CH_BOARD_REVISION_DCP405_R1B1) { + if (slot.moduleRevision == MODULE_REVISION_DCP405_R1B1) { ioexp.changeBit(IOExpander::DCP405_IO_BIT_OUT_CURRENT_RANGE_500MA, false); } @@ -480,14 +595,14 @@ struct Channel : ChannelInterface { // 5A // DebugTrace("CH%d: Switched to 5A range", channel.channelIndex + 1); ioexp.changeBit(IOExpander::DCP405_IO_BIT_OUT_CURRENT_RANGE_5A, true); - ioexp.changeBit(channel.boardRevision == CH_BOARD_REVISION_DCP405_R1B1 ? + ioexp.changeBit(slot.moduleRevision == MODULE_REVISION_DCP405_R1B1 ? IOExpander::DCP405_IO_BIT_OUT_CURRENT_RANGE_50MA : IOExpander::DCP405_R2B5_IO_BIT_OUT_CURRENT_RANGE_50MA, false); // calculateNegligibleAdcDiffForCurrent(); } else { // 50mA // DebugTrace("CH%d: Switched to 50mA range", channel.channelIndex + 1); - ioexp.changeBit(channel.boardRevision == CH_BOARD_REVISION_DCP405_R1B1 ? + ioexp.changeBit(slot.moduleRevision == MODULE_REVISION_DCP405_R1B1 ? IOExpander::DCP405_IO_BIT_OUT_CURRENT_RANGE_50MA : IOExpander::DCP405_R2B5_IO_BIT_OUT_CURRENT_RANGE_50MA, true); ioexp.changeBit(IOExpander::DCP405_IO_BIT_OUT_CURRENT_RANGE_5A, false); @@ -497,7 +612,7 @@ struct Channel : ChannelInterface { adc.start(ADC_DATA_TYPE_U_MON); } else { ioexp.changeBit(IOExpander::DCP405_IO_BIT_OUT_CURRENT_RANGE_5A, true); - ioexp.changeBit(channel.boardRevision == CH_BOARD_REVISION_DCP405_R1B1 ? + ioexp.changeBit(slot.moduleRevision == MODULE_REVISION_DCP405_R1B1 ? IOExpander::DCP405_IO_BIT_OUT_CURRENT_RANGE_50MA : IOExpander::DCP405_R2B5_IO_BIT_OUT_CURRENT_RANGE_50MA, false); } diff --git a/src/eez/modules/dcpX05/dac.cpp b/src/eez/modules/dcpX05/dac.cpp index b946ec370..b77845104 100644 --- a/src/eez/modules/dcpX05/dac.cpp +++ b/src/eez/modules/dcpX05/dac.cpp @@ -77,7 +77,7 @@ bool DigitalAnalogConverter::test(IOExpander &ioexp, AnalogDigitalConverter &adc channel.calibrationEnableNoEvent(false); // disable OE on channel - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { + if (channel.params.features & CH_FEATURE_HW_OVP) { // OVP has to be disabled before OE deactivation ioexp.changeBit(IOExpander::DCP405_IO_BIT_OUT_OVP_ENABLE, false); } @@ -150,7 +150,7 @@ void DigitalAnalogConverter::setVoltage(float value) { Channel &channel = Channel::getBySlotIndex(slotIndex); #if defined(EEZ_PLATFORM_STM32) - set(DATA_BUFFER_B, remap(value, channel.params->U_MIN, (float)DAC_MIN, channel.params->U_MAX, (float)DAC_MAX)); + set(DATA_BUFFER_B, remap(value, channel.params.U_MIN, (float)DAC_MIN, channel.params.U_MAX, (float)DAC_MAX)); #endif #if defined(EEZ_PLATFORM_SIMULATOR) @@ -165,7 +165,7 @@ void DigitalAnalogConverter::setDacVoltage(uint16_t value) { #if defined(EEZ_PLATFORM_SIMULATOR) Channel &channel = Channel::getBySlotIndex(slotIndex); - g_uSet[channel.channelIndex] = remap(value, (float)DAC_MIN, channel.params->U_MIN, (float)DAC_MAX, channel.params->U_MAX); + g_uSet[channel.channelIndex] = remap(value, (float)DAC_MIN, channel.params.U_MIN, (float)DAC_MAX, channel.params.U_MAX); #endif } @@ -173,7 +173,7 @@ void DigitalAnalogConverter::setCurrent(float value) { Channel &channel = Channel::getBySlotIndex(slotIndex); #if defined(EEZ_PLATFORM_STM32) - set(DATA_BUFFER_A, remap(value, channel.params->I_MIN, (float)DAC_MIN, channel.getDualRangeMax(), (float)DAC_MAX)); + set(DATA_BUFFER_A, remap(value, channel.params.I_MIN, (float)DAC_MIN, channel.getDualRangeMax(), (float)DAC_MAX)); #endif #if defined(EEZ_PLATFORM_SIMULATOR) @@ -188,7 +188,7 @@ void DigitalAnalogConverter::setDacCurrent(uint16_t value) { #if defined(EEZ_PLATFORM_SIMULATOR) Channel &channel = Channel::getBySlotIndex(slotIndex); - g_iSet[channel.channelIndex] = remap(value, (float)DAC_MIN, channel.params->I_MIN, (float)DAC_MAX, channel.getDualRangeMax()); + g_iSet[channel.channelIndex] = remap(value, (float)DAC_MIN, channel.params.I_MIN, (float)DAC_MAX, channel.getDualRangeMax()); #endif } diff --git a/src/eez/modules/dcpX05/ioexp.cpp b/src/eez/modules/dcpX05/ioexp.cpp index 8b95bb49d..6bb991c2f 100644 --- a/src/eez/modules/dcpX05/ioexp.cpp +++ b/src/eez/modules/dcpX05/ioexp.cpp @@ -59,18 +59,27 @@ static const uint8_t DCP505_REG_VALUE_IODIRA = 0B00011111; // pins 0, 1, 2, 3, static const uint8_t DCP505_REG_VALUE_IODIRB = 0B11110000; // static const uint8_t DCP405_REG_VALUE_IODIRA = 0B00011111; // pins 0, 1, 2, 3 and 4 are inputs (set to 1) +static const uint8_t DCP405_R2B5_REG_VALUE_IODIRA = 0B00111111; // pins 0, 1, 2, 3, 4 and 5 are inputs (set to 1) static const uint8_t DCP405_REG_VALUE_IODIRB = 0B00000000; // -static const uint8_t DCP405_R2B5_REG_VALUE_IODIRA = 0B00111111; // pins 0, 1, 2, 3, 4 and 5 are inputs (set to 1) +static const uint8_t DCP405B_REG_VALUE_IODIRA = 0B00011111; // pins 0, 1, 2, 3 and 4 are inputs (set to 1) +static const uint8_t DCP405B_REG_VALUE_IODIRB = 0B00000000; // static const uint8_t REG_VALUE_IPOLA = 0B00000000; // no pin is inverted static const uint8_t REG_VALUE_IPOLB = 0B00000000; // no pin is inverted -static const uint8_t REG_VALUE_GPINTENA = 0B00100000; // enable interrupt for HW OVP Fault + +static const uint8_t REG_VALUE_GPINTENA = 0B00000000; // no interrupts +static const uint8_t DCP405_REG_VALUE_GPINTENA = 0B00100000; // enable interrupt for HW OVP Fault static const uint8_t REG_VALUE_GPINTENB = 0B00000000; // no interrupts -static const uint8_t REG_VALUE_DEFVALA = 0B00100000; // default value for HW OVP Fault is 1 + +static const uint8_t REG_VALUE_DEFVALA = 0B00000000; // +static const uint8_t DCP405_REG_VALUE_DEFVALA = 0B00100000; // default value for HW OVP Fault is 1 static const uint8_t REG_VALUE_DEFVALB = 0B00000000; // -static const uint8_t REG_VALUE_INTCONA = 0B00100000; // compare HW OVP Fault value with default value + +static const uint8_t REG_VALUE_INTCONA = 0B00000000; +static const uint8_t DCP405_REG_VALUE_INTCONA = 0B00100000; // compare HW OVP Fault value with default value static const uint8_t REG_VALUE_INTCONB = 0B00000000; // + static const uint8_t REG_VALUE_IOCON = 0B00100000; // sequential operation disabled, hw addressing disabled static const uint8_t REG_VALUE_GPPUA = 0B00100001; // pull up with 100K static const uint8_t REG_VALUE_GPPUB = 0B00000000; // @@ -81,6 +90,9 @@ static const uint8_t DCP505_REG_VALUE_GPIOB = 0B00000001; // DP is OFF static const uint8_t DCP405_REG_VALUE_GPIOA = 0B00000000; // static const uint8_t DCP405_REG_VALUE_GPIOB = 0B00000001; // DP is OFF +static const uint8_t DCP405B_REG_VALUE_GPIOA = 0B00000000; // +static const uint8_t DCP405B_REG_VALUE_GPIOB = 0B00000000; // + static const uint8_t REG_VALUES[] = { REG_IODIRA, DCP505_REG_VALUE_IODIRA, 1, REG_IODIRB, DCP505_REG_VALUE_IODIRB, 1, @@ -98,31 +110,94 @@ static const uint8_t REG_VALUES[] = { REG_GPIOA, DCP505_REG_VALUE_GPIOA, 0, REG_GPIOB, DCP505_REG_VALUE_GPIOB, 0, }; + +static const int REG_IODIRA_INDEX = 0; +static const int REG_IODIRB_INDEX = 1; #endif //////////////////////////////////////////////////////////////////////////////// +#if defined(EEZ_PLATFORM_STM32) +uint8_t IOExpander::getRegValue(int i) { + uint8_t reg = REG_VALUES[3 * i]; + uint8_t value = REG_VALUES[3 * i + 1]; + + auto &slot = g_slots[slotIndex]; + + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP405) { + if (reg == REG_IODIRA) { + if (slot.moduleRevision < MODULE_REVISION_DCP405_R2B5) { + value = DCP405_REG_VALUE_IODIRA; + } else { + value = DCP405_R2B5_REG_VALUE_IODIRA; + } + } else if (reg == REG_IODIRB) { + value = DCP405_REG_VALUE_IODIRB; + } else if (reg == REG_GPIOA) { + value = DCP405_REG_VALUE_GPIOA; + } else if (reg == REG_GPIOB) { + value = DCP405_REG_VALUE_GPIOB; + } else if (reg == REG_GPINTENA) { + value = DCP405_REG_VALUE_GPINTENA; + } else if (reg == REG_DEFVALA) { + value = DCP405_REG_VALUE_DEFVALA; + } else if (reg == REG_INTCONA) { + value = DCP405_REG_VALUE_INTCONA; + } + } else if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP405B) { + if (reg == REG_IODIRA) { + value = DCP405B_REG_VALUE_IODIRA; + } else if (reg == REG_IODIRB) { + value = DCP405B_REG_VALUE_IODIRB; + } else if (reg == REG_GPIOA) { + value = DCP405B_REG_VALUE_GPIOA; + } else if (reg == REG_GPIOB) { + value = DCP405B_REG_VALUE_GPIOB; + } + } + + return value; +} +#endif + void IOExpander::init() { #if defined(EEZ_PLATFORM_STM32) - Channel &channel = Channel::getBySlotIndex(slotIndex); + auto &slot = g_slots[slotIndex]; - gpioOutputPinsMask = - (1 << IO_BIT_OUT_DP_ENABLE) | - (1 << IO_BIT_OUT_OUTPUT_ENABLE) | - (1 << IO_BIT_OUT_REMOTE_SENSE) | - (1 << IO_BIT_OUT_REMOTE_PROGRAMMING); - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { - gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_CURRENT_RANGE_500MA; - gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_CURRENT_RANGE_5A; - gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_OVP_ENABLE; - gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_OE_UNCOUPLED_LED; - gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_OE_COUPLED_LED; - if (channel.boardRevision == CH_BOARD_REVISION_DCP405_R1B1) { + gpioOutputPinsMask = 0; + + if (g_slots[slotIndex].moduleInfo->moduleType == MODULE_TYPE_DCP405) { + gpioOutputPinsMask |= 1 << IO_BIT_OUT_DP_ENABLE; + gpioOutputPinsMask |= 1 << IO_BIT_OUT_OUTPUT_ENABLE; + gpioOutputPinsMask |= 1 << IO_BIT_OUT_REMOTE_SENSE; + gpioOutputPinsMask |= 1 << IO_BIT_OUT_REMOTE_PROGRAMMING; + + if (slot.moduleRevision < MODULE_REVISION_DCP405_R2B5) { gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_CURRENT_RANGE_50MA; + gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_CURRENT_RANGE_500MA; } else { gpioOutputPinsMask |= 1 << DCP405_R2B5_IO_BIT_OUT_CURRENT_RANGE_50MA; } + + gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_CURRENT_RANGE_5A; + + gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_OVP_ENABLE; + gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_OE_UNCOUPLED_LED; + gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_OE_COUPLED_LED; + } else if (g_slots[slotIndex].moduleInfo->moduleType == MODULE_TYPE_DCP405B) { + gpioOutputPinsMask |= 1 << IO_BIT_OUT_OUTPUT_ENABLE; + + gpioOutputPinsMask |= 1 << DCP405_R2B5_IO_BIT_OUT_CURRENT_RANGE_50MA; + gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_CURRENT_RANGE_5A; + + gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_OE_UNCOUPLED_LED; + gpioOutputPinsMask |= 1 << DCP405_IO_BIT_OUT_OE_COUPLED_LED; } else { + // DCP505 + gpioOutputPinsMask |= 1 << IO_BIT_OUT_DP_ENABLE; + gpioOutputPinsMask |= 1 << IO_BIT_OUT_OUTPUT_ENABLE; + gpioOutputPinsMask |= 1 << IO_BIT_OUT_REMOTE_SENSE; + gpioOutputPinsMask |= 1 << IO_BIT_OUT_REMOTE_PROGRAMMING; gpioOutputPinsMask |= 1 << DCP505_IO_BIT_OUT_OVP_ENABLE; gpioOutputPinsMask |= 1 << DCP505_IO_BIT_OUT_OE_UNCOUPLED_LED; gpioOutputPinsMask |= 1 << DCP505_IO_BIT_OUT_OE_COUPLED_LED; @@ -131,24 +206,7 @@ void IOExpander::init() { const uint8_t N_REGS = sizeof(REG_VALUES) / 3; for (int i = 0; i < N_REGS; i++) { uint8_t reg = REG_VALUES[3 * i]; - uint8_t value = REG_VALUES[3 * i + 1]; - - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { - if (reg == REG_IODIRA) { - if (channel.boardRevision == CH_BOARD_REVISION_DCP405_R1B1) { - value = DCP405_REG_VALUE_IODIRA; - } else { - value = DCP405_R2B5_REG_VALUE_IODIRA; - } - } else if (reg == REG_IODIRB) { - value = DCP405_REG_VALUE_IODIRB; - } else if (reg == REG_GPIOA) { - value = DCP405_REG_VALUE_GPIOA; - } else if (reg == REG_GPIOB) { - value = DCP405_REG_VALUE_GPIOB; - } - } - + uint8_t value = getRegValue(i); write(reg, value); } #endif @@ -161,6 +219,7 @@ void IOExpander::init() { bool IOExpander::test() { #if defined(EEZ_PLATFORM_STM32) Channel &channel = Channel::getBySlotIndex(slotIndex); + auto &slot = g_slots[slotIndex]; channel.flags.powerOk = 1; @@ -169,23 +228,9 @@ bool IOExpander::test() { if (REG_VALUES[3 * i + 2]) { uint8_t reg = REG_VALUES[3 * i]; uint8_t value = read(reg); - - uint8_t expectedValue = REG_VALUES[3 * i + 1]; - - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { - if (reg == REG_IODIRA) { - if (channel.boardRevision == CH_BOARD_REVISION_DCP405_R1B1) { - expectedValue = DCP405_REG_VALUE_IODIRA; - } else { - expectedValue = DCP405_R2B5_REG_VALUE_IODIRA; - } - } else if (reg == REG_IODIRB) { - expectedValue = DCP405_REG_VALUE_IODIRB; - } - } - + uint8_t expectedValue = getRegValue(i); if (value != expectedValue) { - DebugTrace("Ch%d IO expander reg check failure: reg=%d, expected=%d, got=%d", channel.channelIndex + 1, (int)REG_VALUES[3 * i], (int)expectedValue, (int)value); + DebugTrace("Ch%d IO expander reg check failure: reg=%d, expected=%d, got=%d", slot.channelIndex + 1, (int)REG_VALUES[3 * i], (int)expectedValue, (int)value); g_testResult = TEST_FAILED; break; @@ -196,7 +241,7 @@ bool IOExpander::test() { readGpio(); if (g_testResult == TEST_FAILED) { - generateError(SCPI_ERROR_CH1_IOEXP_TEST_FAILED + channel.channelIndex); + generateError(SCPI_ERROR_CH1_IOEXP_TEST_FAILED + slot.channelIndex); channel.flags.powerOk = 0; } else { g_testResult = TEST_OK; @@ -204,8 +249,8 @@ bool IOExpander::test() { #if !CONF_SKIP_PWRGOOD_TEST channel.flags.powerOk = testBit(IO_BIT_IN_PWRGOOD); if (!channel.flags.powerOk) { - DebugTrace("Ch%d power fault", channel.channelIndex + 1); - generateError(SCPI_ERROR_CH1_FAULT_DETECTED - channel.channelIndex); + DebugTrace("Ch%d power fault", slot.channelIndex + 1); + generateError(SCPI_ERROR_CH1_FAULT_DETECTED + slot.channelIndex); } #endif } @@ -222,24 +267,10 @@ bool IOExpander::test() { #if defined(EEZ_PLATFORM_STM32) void IOExpander::reinit() { - Channel &channel = Channel::getBySlotIndex(slotIndex); - const uint8_t N_REGS = sizeof(REG_VALUES) / 3; for (int i = 0; i < N_REGS; i++) { uint8_t reg = REG_VALUES[3 * i]; - uint8_t value = REG_VALUES[3 * i + 1]; - - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { - if (reg == REG_IODIRA) { - if (channel.boardRevision == CH_BOARD_REVISION_DCP405_R1B1) { - value = DCP405_REG_VALUE_IODIRA; - } else { - value = DCP405_R2B5_REG_VALUE_IODIRA; - } - } else if (reg == REG_IODIRB) { - value = DCP405_REG_VALUE_IODIRB; - } - } + uint8_t value = getRegValue(i); if (reg == REG_GPIOA) { value = (uint8_t)gpioWritten; @@ -254,16 +285,16 @@ void IOExpander::reinit() { void IOExpander::tick(uint32_t tick_usec) { #if defined(EEZ_PLATFORM_STM32) - Channel &channel = Channel::getBySlotIndex(slotIndex); + auto &slot = g_slots[slotIndex]; readGpio(); uint8_t iodira = read(REG_IODIRA); if (iodira == 0xFF || (gpio & gpioOutputPinsMask) != gpioWritten) { if (iodira == 0xFF) { - event_queue::pushEvent(event_queue::EVENT_ERROR_CH1_IOEXP_RESET_DETECTED + channel.channelIndex); + event_queue::pushEvent(event_queue::EVENT_ERROR_CH1_IOEXP_RESET_DETECTED + slot.channelIndex); } else { - event_queue::pushEvent(event_queue::EVENT_ERROR_CH1_IOEXP_FAULT_MATCH_DETECTED + channel.channelIndex); + event_queue::pushEvent(event_queue::EVENT_ERROR_CH1_IOEXP_FAULT_MATCH_DETECTED + slot.channelIndex); } reinit(); @@ -281,7 +312,7 @@ void IOExpander::tick(uint32_t tick_usec) { gpio &= ~(1 << IOExpander::IO_BIT_IN_PWRGOOD); } - if (channel.getFeatures() & CH_FEATURE_RPOL) { + if (channel.params.features & CH_FEATURE_RPOL) { if (!simulator::getRPol(channel.channelIndex)) { gpio |= 1 << IOExpander::IO_BIT_IN_RPOL; } else { @@ -307,9 +338,9 @@ void IOExpander::tick(uint32_t tick_usec) { int IOExpander::getBitDirection(int bit) { uint8_t dir; if (bit < 8) { - dir = g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405 ? DCP405_REG_VALUE_IODIRA : DCP505_REG_VALUE_IODIRA; + dir = getRegValue(REG_IODIRA_INDEX); } else { - dir = g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405 ? DCP405_REG_VALUE_IODIRB : DCP505_REG_VALUE_IODIRB; + dir = getRegValue(REG_IODIRB_INDEX); bit -= 8; } return dir & (1 << bit) ? 1 : 0; @@ -323,7 +354,8 @@ bool IOExpander::testBit(int io_bit) { #if defined(EEZ_PLATFORM_STM32) bool IOExpander::isAdcReady() { // ready = !HAL_GPIO_ReadPin(SPI2_IRQ_GPIO_Port, SPI2_IRQ_Pin); - return !testBit(g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405 ? DCP405_IO_BIT_IN_ADC_DRDY : DCP505_IO_BIT_IN_ADC_DRDY); + auto &slot = g_slots[slotIndex]; + return !testBit(slot.moduleInfo->moduleType == MODULE_TYPE_DCP405 || slot.moduleInfo->moduleType == MODULE_TYPE_DCP405B ? DCP405_IO_BIT_IN_ADC_DRDY : DCP505_IO_BIT_IN_ADC_DRDY); } #endif diff --git a/src/eez/modules/dcpX05/ioexp.h b/src/eez/modules/dcpX05/ioexp.h index ba2157c12..a96e45feb 100644 --- a/src/eez/modules/dcpX05/ioexp.h +++ b/src/eez/modules/dcpX05/ioexp.h @@ -51,9 +51,11 @@ class IOExpander { static const uint8_t DCP405_IO_BIT_IN_ADC_DRDY = 4; static const uint8_t DCP405_IO_BIT_OUT_CURRENT_RANGE_50MA = 5; - static const uint8_t DCP405_R2B5_IO_BIT_IN_OVP_FAULT = 5; // active low static const uint8_t DCP405_IO_BIT_OUT_CURRENT_RANGE_500MA = 6; + + static const uint8_t DCP405_R2B5_IO_BIT_IN_OVP_FAULT = 5; // active low static const uint8_t DCP405_R2B5_IO_BIT_OUT_CURRENT_RANGE_50MA = 6; + static const uint8_t DCP405_IO_BIT_OUT_CURRENT_RANGE_5A = 7; static const uint8_t DCP405_IO_BIT_OUT_OVP_ENABLE = 12; @@ -86,6 +88,8 @@ class IOExpander { uint16_t gpioWritten; uint16_t gpioOutputPinsMask; + uint8_t getRegValue(int i); + void reinit(); void readGpio(); uint8_t read(uint8_t reg); diff --git a/src/eez/platform/simulator/front_panel.cpp b/src/eez/platform/simulator/front_panel.cpp index f4035ff13..ac11fd90a 100644 --- a/src/eez/platform/simulator/front_panel.cpp +++ b/src/eez/platform/simulator/front_panel.cpp @@ -68,9 +68,11 @@ void data_main_app_view(DataOperationEnum operation, Cursor &cursor, Value &valu int getSlotView(int channelIndex) { int slotIndex = psu::Channel::get(channelIndex).slotIndex; - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { + if (g_slots[slotIndex].moduleInfo->moduleType == MODULE_TYPE_DCP405) { return PAGE_ID_DCP405_FRONT_PANEL; - } else if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCM220) { + } else if (g_slots[slotIndex].moduleInfo->moduleType == MODULE_TYPE_DCP405B) { + return PAGE_ID_DCP405B_FRONT_PANEL; + } else if (g_slots[slotIndex].moduleInfo->moduleType == MODULE_TYPE_DCM220) { return PAGE_ID_DCM220_FRONT_PANEL; } else { return PAGE_ID_FRONT_PANEL_EMPTY_SLOT; diff --git a/src/eez/platform/stm32/spi.cpp b/src/eez/platform/stm32/spi.cpp index 781f6d650..41c1deec4 100644 --- a/src/eez/platform/stm32/spi.cpp +++ b/src/eez/platform/stm32/spi.cpp @@ -85,7 +85,9 @@ void deselectB(uint8_t slotIndex) { void select(uint8_t slotIndex, int chip) { taskENTER_CRITICAL(); - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCM220) { + auto &slot = g_slots[slotIndex]; + + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCM220) { selectA(slotIndex); return; } @@ -96,7 +98,7 @@ void select(uint8_t slotIndex, int chip) { // __HAL_SPI_ENABLE(spiHandle[slotIndex]); - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP405 || slot.moduleInfo->moduleType == MODULE_TYPE_DCP405B) { if (chip == CHIP_DAC) { // 00 selectA(slotIndex); @@ -134,9 +136,10 @@ void select(uint8_t slotIndex, int chip) { } void deselect(uint8_t slotIndex) { - if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCM220) { + auto &slot = g_slots[slotIndex]; + if (slot.moduleInfo->moduleType == MODULE_TYPE_DCM220) { deselectA(slotIndex); - } else if (g_slots[slotIndex].moduleType == MODULE_TYPE_DCP405) { + } else if (slot.moduleInfo->moduleType == MODULE_TYPE_DCP405 || slot.moduleInfo->moduleType == MODULE_TYPE_DCP405B) { // 01 ADC deselectA(slotIndex); selectB(slotIndex);