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);