Готов механизм транзакций на роботе с пачкой команд

This commit is contained in:
VladislavOstapov 2022-12-03 00:44:56 +03:00
parent 8f19d1b277
commit 2e4a560f74
7 changed files with 401 additions and 422 deletions

View File

@ -1,2 +1,9 @@
# sdp-scheduler # sdp-scheduler
## Ограничения программы
* нельзя отключить сразу все зоны цинкования или травления
* коррекцию нужно описать так, чтобы не произошло ничего страшного
* время скапывания нельзя поставить больше 200 секунд (в текущей реализации можно до 8К секунд)
* время каждого процесса не больше чем 32К секунд (можно уменьшить до 8К для упрощения логики)

View File

@ -174,7 +174,7 @@ static void open_socket() {
} }
} }
extern "C" struct barrel makeBarrel(short flags, short zone, short timer, short process); extern "C" struct barrel makeBarrel(short flags, short zone, short timer);
int main() { int main() {
open_socket(); open_socket();
@ -185,7 +185,7 @@ int main() {
// b.zone = (short) abs(random() % 20); // b.zone = (short) abs(random() % 20);
// } // }
barrels[5] = makeBarrel(1, 10, 3, PROCESS_WASHING_3A); barrels[5] = makeBarrel(1, 10, 3);
current_tic = 0; current_tic = 0;
const char* message = nullptr; const char* message = nullptr;

View File

@ -16,6 +16,8 @@ extern char schedulerLoadButton1;
extern char schedulerLoadButton2; extern char schedulerLoadButton2;
extern char schedulerOneRobotMode; extern char schedulerOneRobotMode;
extern short etching_zone, galvanizing_zone;
void scheduler_main(); void scheduler_main();
@ -49,8 +51,8 @@ extern char button_unload_remove;
// Переменные, которые надо добавить в C Global variables // Переменные, которые надо добавить в C Global variables
// код для роботов // код для роботов
extern robot_code robot1_code; extern struct robot_code robot1_code;
extern robot_code robot2_code; extern struct robot_code robot2_code;
// lock-зоны, нельзя двигаться за них и за робота // lock-зоны, нельзя двигаться за них и за робота

235
robot.cpp
View File

@ -19,7 +19,7 @@ static short get_barrel(char robot_id) {
} }
// true означает что движение закончено // true означает что движение закончено
static bool robot_move(robot_regs& r, int target, char robot_id) { static bool robot_move(robot_regs& r, short target, char robot_id) {
// после перемещения мы явно в точной позиции // после перемещения мы явно в точной позиции
if (robot_id == 1) { if (robot_id == 1) {
robot1_offset_pos = false; robot1_offset_pos = false;
@ -39,167 +39,21 @@ static bool robot_move(robot_regs& r, int target, char robot_id) {
return false; return false;
} }
static void emulate_robot(robot_cmd& cmd, robot_regs& r, char robot_id) {
auto barrel = get_barrel(robot_id);
switch (cmd.cmd) {
case 1:
// команда просто уехать
switch (cmd.step) {
case 0:
// двигаемся в сторону цели
if (robot_move(r, cmd.args[0], robot_id)) {
cmd.step++;
}
break;
default: static void emulate_robot(robot_code &code, robot_regs& r, char robot_id) {
cmd.cmd = 0; if (code.PC < 0) {
cmd.step = 0;
}
break;
case 2:
// команда взять барабан и увезти его куда положено
switch (cmd.step) {
case 0:
// двигаемся в сторону барабана
if (robot_move(r, cmd.args[0], robot_id)) {
cmd.step++;
}
break;
case 1:
// поднимаем траверсу
r.mz.is_up = 1;
if (barrel != -1) {
barrels[barrel].flags.is_up = 1;
}
cmd.step++;
break;
case 2:
// двигаемся в сторону выгрузки
if (robot_move(r, cmd.args[1], robot_id)) {
cmd.step++;
}
break;
case 3:
// опускаем траверсу
r.mz.is_up = 0;
if (barrel != -1) {
barrels[barrel].flags.is_up = 0;
}
// ну и тут же конец
default:
cmd.cmd = 0;
cmd.step = 0;
}
break;
case 3:
// команда пассивация
switch (cmd.step) {
case 0:
// двигаемся в 18 зону
if (robot_move(r, 18, robot_id)) {
cmd.step++;
}
break;
case 1:
// поднимаем траверсу
r.mz.is_up = 1;
if (barrel != -1) {
barrels[barrel].flags.is_up = 1;
}
cmd.step++;
break;
case 2:
// двигаемся в 19 зону
if (robot_move(r, 19, robot_id)) {
cmd.step++;
}
break;
case 3:
// опускаем траверсу
r.mz.is_up = 0;
if (barrel != -1) {
barrels[barrel].flags.is_up = 0;
barrels[barrel].software_timer = barrels[barrel].time_passivation;
}
cmd.step++;
break;
case 4:
// ждем пока барабан отстоит таймер в пассивации
if (barrel != -1) {
if (barrels[barrel].software_timer <= 0) {
cmd.step++;
}
}
break;
case 5:
// поднимаем траверсу
r.mz.is_up = 1;
if (barrel != -1) {
barrels[barrel].flags.is_up = 1;
}
cmd.step++;
break;
case 6:
// двигаемся в 20 зону
if (robot_move(r, 20, robot_id)) {
cmd.step++;
}
break;
case 7:
// опускаем траверсу
r.mz.is_up = 0;
if (barrel != -1) {
barrels[barrel].flags.is_up = 0;
// и тут же ставим время промывки барабана
barrels[barrel].software_timer = barrels[barrel].time_washing_4a;
}
// ну и тут же конец
default:
cmd.cmd = 0;
cmd.step = 0;
}
break;
default:
cmd.cmd = 0;
cmd.step = 0;
}
if (r.mz.is_up) {
if (barrel != -1) {
barrels[barrel].zone = r.dx.current_zone;
}
}
}
static void emulate_robot_v2(robot_code* code, robot_regs& r, char robot_id) {
if (code->PC < 0) {
return; return;
} }
const short cmd_arg = code->code[code->PC] & (~ROBOT_CMD_MASK); const short cmd_arg = code.code[code.PC] & (~ROBOT_CMD_MASK);
switch (code->code[code->PC] & ROBOT_CMD_MASK) { switch (code.code[code.PC] & ROBOT_CMD_MASK) {
case ROBOT_CMD_MOVE_TO_ZONE: case ROBOT_CMD_MOVE_TO_ZONE:
// двигаемся в сторону цели // двигаемся в сторону цели
if (robot_move(r, cmd_arg, robot_id)) { if (robot_move(r, cmd_arg & (~ROBOT_WITH_BARREL), robot_id)) {
code->PC++; code.PC++;
}
if (cmd_arg & ROBOT_WITH_BARREL && code.barrel_id >= 0) {
barrels[code.barrel_id].zone = r.dx.current_zone;
} }
break; break;
@ -209,38 +63,42 @@ static void emulate_robot_v2(robot_code* code, robot_regs& r, char robot_id) {
} else { } else {
robot2_offset_pos = true; robot2_offset_pos = true;
} }
code->PC++; code.PC++;
break; break;
case ROBOT_CMD_UP: case ROBOT_CMD_UP:
if (code->barrel_id >= 0) { if (code.barrel_id >= 0 && cmd_arg) {
barrels[code->barrel_id].flags.is_up = true; // не давать ехать перед тем как истечет таймер
if (barrels[code.barrel_id].software_timer > 0) {
break;
}
barrels[code.barrel_id].flags.is_up = true;
r.mz.is_up = 1; r.mz.is_up = 1;
} }
code->PC++; code.PC++;
break; break;
// в эмуляторе не важно где я, поэтому тут обе команды вниз обрабатываются одинаково // в эмуляторе не важно где я, поэтому тут обе команды вниз обрабатываются одинаково
case ROBOT_CMD_DOWN: case ROBOT_CMD_DOWN:
case ROBOT_CMD_DOWN_2: case ROBOT_CMD_DOWN_2:
if (code->barrel_id >= 0) { if (code.barrel_id >= 0 && cmd_arg) {
barrels[code->barrel_id].flags.is_up = false; barrels[code.barrel_id].flags.is_up = false;
r.mz.is_up = 0; r.mz.is_up = 0;
} }
code->PC++; code.PC++;
break; break;
case ROBOT_CMD_WAIT: case ROBOT_CMD_WAIT:
std::cout << "robot " << robot_id << " wait " << cmd_arg << " secs..." << std::endl; std::cout << "robot " << robot_id << " wait " << cmd_arg << " secs..." << std::endl;
code->PC++; code.PC++;
break; break;
case ROBOT_CMD_TMR_SET: case ROBOT_CMD_TMR_SET:
if (code->barrel_id >= 0) { if (code.barrel_id >= 0) {
barrels[code->barrel_id].software_timer = code->code[code->PC + 1]; barrels[code.barrel_id].software_timer = code.code[code.PC + 1];
r.mz.is_up = 0; r.mz.is_up = 0;
} }
code->PC += 2; code.PC += 2;
break; break;
case ROBOT_CMD_SET_LOCK_ZONE: case ROBOT_CMD_SET_LOCK_ZONE:
@ -249,31 +107,46 @@ static void emulate_robot_v2(robot_code* code, robot_regs& r, char robot_id) {
} else { } else {
robot2_lock_zone = cmd_arg; robot2_lock_zone = cmd_arg;
} }
code->PC += 2; code.PC += 2;
break; break;
case ROBOT_CMD_CORRECT_X: case ROBOT_CMD_CORRECT_AXIS:
std::cout << "robot " << robot_id << " correct axis X..." << std::endl; if (cmd_arg == ROBOT_AXIS_X) {
r.dz.current_zone = 0; std::cout << "robot " << robot_id << " correct axis X..." << std::endl;
r.mx.correct_status = true; r.dz.current_zone = 0;
code->PC++; r.mx.correct_status = true;
} else if (cmd_arg == ROBOT_AXIS_Z) {
std::cout << "robot " << robot_id << " correct axis Z..." << std::endl;
r.mz.is_up = true;
r.mz.correct_status = true;
} else {
std::cout << "ERROR: R" << robot_id << " AXIS CORRECT - INCORRECT ARGUMENT VALUE " << cmd_arg << std::endl;
}
code.PC++;
break; break;
case ROBOT_CMD_CORRECT_Z: case ROBOT_CMD_INC_ZONE:
std::cout << "robot " << robot_id << " correct axis Z..." << std::endl; if (cmd_arg == ROBOT_ZONE_ETCH) {
r.mz.is_up = true; std::cout << "robot " << robot_id << " increment etching..." << std::endl;
r.mz.correct_status = true; etching_zone = (etching_zone + 1) & 1;
code->PC++; } else if (cmd_arg == ROBOT_ZONE_GAL) {
std::cout << "robot " << robot_id << " increment galvanic..." << std::endl;
galvanizing_zone = (galvanizing_zone + 1) & 0x07;
} else {
std::cout << "ERROR: R" << robot_id << " INCREMENT ZONE - INCORRECT ARGUMENT VALUE " << cmd_arg << std::endl;
}
code.PC++;
break; break;
case ROBOT_CMD_END: case ROBOT_CMD_END:
default: default:
code->PC = -1; code.PC = -1;
} }
} }
void robot_main() { void robot_main() {
emulate_robot(robot1_cmd, robot1, 1); emulate_robot(robot1_code, robot1, 1);
emulate_robot(robot2_cmd, robot2, 2); emulate_robot(robot2_code, robot2, 2);
} }

View File

@ -46,118 +46,125 @@ short can_move(struct barrel* bar) {
// дальше нужно проверить, что можно передвигать бочку // дальше нужно проверить, что можно передвигать бочку
switch (bar->curr_process) { switch (bar->zone) {
case PROCESS_NONE: case ZONE_LOAD_2:
// загрузка, нужно обезжиривание (зона 2) // загрузка 2, только в нее можно грузить новые барабаны, нужно обезжиривание
if (!zone_is_busy(2)) { if (!zone_is_busy(ZONE_DEGREASING)) {
return 2; return ZONE_DEGREASING;
} }
break; break;
case PROCESS_DEFATTING: case ZONE_DEGREASING:
// обезжиривание, нужна промывка 1А (зона 3) // обезжиривание, нужна промывка 1А
if (!zone_is_busy(3)) { if (!zone_is_busy(ZONE_WASHING_1A)) {
return 3; return ZONE_WASHING_1A;
} }
break; break;
case PROCESS_WASHING_1A: case ZONE_WASHING_1A:
// промывка 1А, нужна промывка 1Б (зона 4) // промывка 1А, нужна промывка 1Б
if (!zone_is_busy(4)) { if (!zone_is_busy(ZONE_WASHING_1B)) {
return 4; return ZONE_WASHING_1B;
} }
break; break;
case PROCESS_WASHING_1B: case ZONE_WASHING_1B:
// промывка 1Б, нужно травление (зоны 5-6) // промывка 1Б, нужно травление (зоны 5-6)
if (!zone_is_busy(5 + etching_zone)) { if (!zone_is_busy(ZONE_ETCHING_1 + etching_zone)) {
return 5 + etching_zone; return ZONE_ETCHING_1 + etching_zone;
} }
break; break;
case PROCESS_ETCHING: case ZONE_ETCHING_1:
// травление, нужна промывка 2А (зона 7) case ZONE_ETCHING_2:
if (!zone_is_busy(7)) { // травление, нужна промывка 2А
return 7; if (!zone_is_busy(ZONE_WASHING_2A)) {
return ZONE_WASHING_2A;
} }
break; break;
case PROCESS_WASHING_2A: case ZONE_WASHING_2A:
// промывка 2А, нужна промывка 2Б (зона 8) // промывка 2А, нужна промывка 2Б
if (!zone_is_busy(8)) { if (!zone_is_busy(ZONE_WASHING_2B)) {
return 8; return ZONE_WASHING_2B;
} }
break; break;
case PROCESS_WASHING_2B: case ZONE_WASHING_2B:
// промывка 2Б, нужно цинкование (зоны 9-16) // промывка 2Б, нужно цинкование (зоны 9-16)
if (!zone_is_busy(9 + galvanizing_zone)) { if (!zone_is_busy(ZONE_GALVANIZING_1 + galvanizing_zone)) {
return 9 + galvanizing_zone; return ZONE_GALVANIZING_1 + galvanizing_zone;
} }
break; break;
case PROCESS_GALVANIZING: case ZONE_GALVANIZING_1:
case ZONE_GALVANIZING_2:
case ZONE_GALVANIZING_3:
case ZONE_GALVANIZING_4:
case ZONE_GALVANIZING_5:
case ZONE_GALVANIZING_6:
case ZONE_GALVANIZING_7:
case ZONE_GALVANIZING_8:
// цинкование, требуется чтобы в зонах 17-22 было максимум 2 барабана (3 барабана для этой части линии - максимум) // цинкование, требуется чтобы в зонах 17-22 было максимум 2 барабана (3 барабана для этой части линии - максимум)
{ if (!zone_is_busy(ZONE_WASHING_3A)) {
short count = 0; short count = 0;
for (short i = 17; i <= 22; i++) { // если зона 17 свободна, то диапазон начнется с
for (short i = ZONE_WASHING_3B; i <= ZONE_UNLOAD; i++) {
if (zone_is_busy(i)) { if (zone_is_busy(i)) {
count++; count++;
} }
} }
if (count < 3 && !zone_is_busy(17)) { if (count < 3) {
return 17; return ZONE_WASHING_3A;
} }
} }
break; break;
case PROCESS_WASHING_3A: case ZONE_WASHING_3A:
// промывка 3А, перекладываем в промывку 3Б (зона 18) // промывка 3А, перекладываем в промывку 3Б
if (!zone_is_busy(18)) { if (!zone_is_busy(ZONE_WASHING_3B)) {
return 18; return ZONE_WASHING_3B;
} }
break; break;
case PROCESS_WASHING_3B: case ZONE_WASHING_3B:
// это перед пассивацией, требует свободную промывку 4А (зона 20) и на всякий случай свободную пассивацию (зона 19) // это перед пассивацией, требует свободную промывку 4А и на всякий случай свободную пассивацию
if (!zone_is_busy(19) && !zone_is_busy(20)) { if (!zone_is_busy(ZONE_PASSIVATION) && !zone_is_busy(ZONE_WASHING_4A)) {
return -2; return ZONE_PASSIVATION;
} }
// это атомарная операция, по идее вносить барабан в пассивацию нельзя // это атомарная операция, по идее вносить барабан в пассивацию нельзя
break; break;
case PROCESS_PASSIVATION:
// процесс пассивации, нужна промывка 4B (зона 21) (потому что сейчас я в 4A) case ZONE_PASSIVATION:
if (!zone_is_busy(21)) { // процесс пассивации, нужна промывка 4A
return 21; // чисто теоретически сюда никогда не попадем, но если вдруг выстрелит пусть будет
if (!zone_is_busy(ZONE_WASHING_4A)) {
return ZONE_WASHING_4A;
} }
break; break;
case PROCESS_WASHING_4B:
// процесс пассивации, нужна промывка 4B (зона 21) (потому что сейчас я в 4A) case ZONE_WASHING_4A:
if (!zone_is_busy(22)) { // промывка 4А, перекладываем в промывку 4Б
return 22; if (!zone_is_busy(ZONE_WASHING_4B)) {
return ZONE_WASHING_4B;
} }
break; break;
case PROCESS_RETURN_1:
case ZONE_WASHING_4B:
// процесс пассивации, нужна промывка 4B (зона 21) (потому что сейчас я в 4A)
if (!zone_is_busy(ZONE_UNLOAD)) {
return ZONE_UNLOAD;
}
break;
case ZONE_UNLOAD:
// последняя промывка, нужно разрешение на выгрузку // последняя промывка, нужно разрешение на выгрузку
if (schedulerUnloadButton) { if (schedulerOneRobotMode) {
// нужно промывку 3б (зона 10) и загрузку 0 // нужно промывку загрузку 0
if (!zone_is_busy(17) || !zone_is_busy(0)) { if (!zone_is_busy(17)) {
return 17; return 17;
} }
} }
break;
case PROCESS_RETURN_2:
if (schedulerOneRobotMode) {
if (!schedulerUnloadButton) {
break;
}
}
// нужна свободная выгрузка
if (!zone_is_busy(0)) {
return 0;
}
break; break;
} }
return -1; return -1;
@ -184,13 +191,12 @@ short get_operation_priority(short barrel_id) {
} }
struct barrel makeBarrel(short flags, short zone, short timer, short process) { struct barrel makeBarrel(short flags, short zone, short timer) {
struct barrel b; struct barrel b;
b.flags.raw_word = flags; b.flags.raw_word = flags;
b.zone = zone; b.zone = zone;
b.software_timer = timer; b.software_timer = timer;
b.curr_process = process; b.time_degreasing = 12;
b.time_defatting = 12;
b.time_washing_1a = 4; b.time_washing_1a = 4;
b.time_washing_1b = 6; b.time_washing_1b = 6;
b.time_etching = 16; b.time_etching = 16;
@ -261,135 +267,236 @@ static short scheduler_find_task(struct scheduler_task* tasks, const short curr_
} }
void schedule_robot_1() { void debug_print_robot1_code() {
static short transaction_state = 0; // printf("INFO: code length is %d\n", cmd_index);
printf("Code for R0, B%d:\n", robot1_code.barrel_id);
for (int i = 0; i < 16; i++) {
const short cmd_arg = robot1_code.code[i] & (~ROBOT_CMD_MASK);
// начало транзакции if ((robot1_code.code[i] & ROBOT_CMD_MASK) == ROBOT_CMD_END) {
printf(" END\n");
if (transaction_state == 0) { break;
// формируем список задач
struct scheduler_task tasks[BARRELS_COUNT];
for (short i = 0; i < BARRELS_COUNT; i++) {
// для каждой задачи:
tasks[i].start_zone = barrels[i].zone;
// определяем можно ли ее выполнить и что вообще нужно выполнить
tasks[i].dest_zone = can_move(barrels + i);
} }
// найти подходящую задачу switch (robot1_code.code[i] & ROBOT_CMD_MASK) {
if (schedulerOneRobotMode) { case ROBOT_CMD_MOVE_TO_ZONE:
short target_task = scheduler_find_task(tasks, robot1.dx.current_zone); printf(" move to zone %d (with barrel: %d)\n", cmd_arg & (~ROBOT_WITH_BARREL), (cmd_arg & ROBOT_WITH_BARREL) != 0);
if (target_task >= 0) {
// создаем транзакцию
if (tasks[target_task].dest_zone == -2) {
// пассивация
printf("Get work: passivate barrel %d\n", target_task);
robot1_cmd.cmd = 3; // пассивация
} else {
printf("Get work: move barrel %d from %d to %d\n",
target_task, tasks[target_task].start_zone, tasks[target_task].dest_zone);
robot1_cmd.cmd = 2; // везем барабан
robot1_cmd.args[0] = tasks[target_task].start_zone;
robot1_cmd.args[1] = tasks[target_task].dest_zone;
}
// barrels[target_barrel].zone = target_zone;
barrels[target_task].flags.robot = 1;
// TODO сделать нормальное переключение зон с учетом тех, что можно отключить
// TODO добавить ограничение - нельзя отключить сразу все зоны цинкования или травления
if (tasks[target_task].dest_zone == 3 || tasks[target_task].dest_zone == 4) {
etching_zone ^= 0x1; // переключаем следующую зону
}
if (tasks[target_task].dest_zone >= 9 && tasks[target_task].dest_zone <= 17) {
galvanizing_zone = (galvanizing_zone + 1) % 8;
}
if (tasks[target_task].start_zone == 22) {
// выгрузка, снимаем кнопку выгрузки
schedulerUnloadButton = 0;
}
transaction_state = 1;
}
} else {
printf("WARMING: support only one robot mode\n");
}
} else if (transaction_state == 1) {
if (robot1_cmd.cmd == 0) {
transaction_state = 2;
}
} else {
// post transaction
short barrel = get_robot_barrel(1);
struct barrel* b = &barrels[barrel];
if (barrel >= 0) {
b->flags.is_up = 0;
b->flags.robot = 0;
if (robot1_cmd.cmd == 2) {
b->zone = robot1_cmd.args[2];
} else if (robot1_cmd.cmd == 3) {
b->zone = 20;
}
}
switch (b->curr_process) {
// case curr: bar->curr_process = next; bar->software_timer = bar->time; break
// после загрузки: ставим обезжиривание
case PROCESS_NONE: b->curr_process = PROCESS_DEFATTING; b->software_timer = b->time_defatting; break;
// после обезжира: ставим промывку 1а
case PROCESS_DEFATTING: b->curr_process = PROCESS_WASHING_1A; b->software_timer = b->time_washing_1a; break;
// после промывки 1а: ставим промывку 1б
case PROCESS_WASHING_1A: b->curr_process = PROCESS_WASHING_1B; b->software_timer = b->time_washing_1b; break;
// после промывки 1а: ставим травление
case PROCESS_WASHING_1B: b->curr_process = PROCESS_ETCHING; b->software_timer = b->time_etching; break;
// после травления: ставим промывку 2а
case PROCESS_ETCHING: b->curr_process = PROCESS_WASHING_2A; b->software_timer = b->time_washing_2a; break;
// после промывки 2а: ставим промывку 2б
case PROCESS_WASHING_2A: b->curr_process = PROCESS_WASHING_2B; b->software_timer = b->time_washing_2b; break;
// после промывки 2б: ставим цинкование
case PROCESS_WASHING_2B: b->curr_process = PROCESS_GALVANIZING; b->software_timer = b->time_galvanizing; break;
// после цинкования: ставим промывку 3а
case PROCESS_GALVANIZING: b->curr_process = PROCESS_WASHING_3A; b->software_timer = b->time_washing_3a; break;
// после промывки 3а: ставим промывку 1б
case PROCESS_WASHING_3A: b->curr_process = PROCESS_WASHING_3B; b->software_timer = b->time_washing_3b; break;
// после промывки 3б: ставим пассивацию
case PROCESS_WASHING_3B: b->curr_process = PROCESS_PASSIVATION; break;
// после пассивации, барабан в промывке 4а: ставим промывку 4б
case PROCESS_PASSIVATION: b->curr_process = PROCESS_WASHING_4B; b->software_timer = b->time_washing_4b; break;
// после промывки 4б: дальше выгрузка
case PROCESS_WASHING_4B:
if (schedulerOneRobotMode) {
b->curr_process = PROCESS_RETURN_2;
} else {
b->curr_process = PROCESS_RETURN_1;
}
b->software_timer = 0;
break; break;
case PROCESS_RETURN_1: b->curr_process = PROCESS_RETURN_2; b->software_timer = 0; break; case ROBOT_CMD_MOVE_OFF:
printf(" move to offset pos\n");
break;
case ROBOT_CMD_UP:
printf(" up (with barrel: %d)\n", (cmd_arg & ROBOT_WITH_BARREL) != 0);
break;
// в эмуляторе не важно где я, поэтому тут обе команды вниз обрабатываются одинаково
case ROBOT_CMD_DOWN:
printf(" down (with barrel: %d)\n", (cmd_arg & ROBOT_WITH_BARREL) != 0);
break;
case ROBOT_CMD_DOWN_2:
printf(" down 2 (with barrel: %d)\n", (cmd_arg & ROBOT_WITH_BARREL) != 0);
break;
case ROBOT_CMD_WAIT:
printf(" wait %d secs\n", cmd_arg);
break;
case ROBOT_CMD_TMR_SET:
printf(" set barrel timer %d secs\n", cmd_arg);
i++;
break;
case ROBOT_CMD_SET_LOCK_ZONE:
printf(" set lock zone %d\n", cmd_arg);
break;
case ROBOT_CMD_CORRECT_AXIS:
if (cmd_arg == ROBOT_AXIS_X) {
printf(" correct axis: X\n");
} else if (cmd_arg == ROBOT_AXIS_Z) {
printf(" correct axis: Z\n");
} else {
printf(" correct axis: INVALID (%d)\n", cmd_arg);
}
break;
case ROBOT_CMD_INC_ZONE:
if (cmd_arg == ROBOT_ZONE_ETCH) {
printf(" increment: etching\n");
} else if (cmd_arg == ROBOT_ZONE_GAL) {
printf(" increment: galvanic\n");
} else {
printf(" increment: INVALID (%d)\n", cmd_arg);
}
break;
case PROCESS_RETURN_2:
default: default:
b->flags.is_exist = 0; printf(" UNKNOWN: 0x(%04X)\n", cmd_arg);
} }
}
}
transaction_state = 0;
void schedule_robot_1() {
// формируем список задач
struct scheduler_task tasks[BARRELS_COUNT];
for (short i = 0; i < BARRELS_COUNT; i++) {
// для каждой задачи:
tasks[i].start_zone = barrels[i].zone;
// определяем можно ли ее выполнить и что вообще нужно выполнить
tasks[i].dest_zone = can_move(barrels + i);
}
// найти подходящую задачу
if (schedulerOneRobotMode) {
short target_task = scheduler_find_task(tasks, robot1.dx.current_zone);
if (target_task >= 0) {
// создаем код транзакции, пока обычный
robot1_code.barrel_id = target_task;
short cmd_index = 0;
// первым делом добавляем команду опустить траверсу
robot1_code.code[cmd_index++] = ROBOT_CMD_DOWN;
if (tasks[target_task].start_zone != ZONE_WASHING_3B) {
if (robot1.dx.current_zone != tasks[target_task].start_zone) {
if (tasks[target_task].start_zone == 22) {
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_OFF;
robot1_code.code[cmd_index++] = ROBOT_CMD_UP;
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_TO_ZONE | 21;
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_OFF;
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_TO_ZONE | 22;
} else {
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_TO_ZONE | tasks[target_task].start_zone;
}
}
robot1_code.code[cmd_index++] = ROBOT_CMD_UP_WITH_BARREL;
// теперь надо определиться с тем, сколько ждать скапывания
switch (tasks[target_task].start_zone) {
case ZONE_DEGREASING:
case ZONE_ETCHING_1:
case ZONE_ETCHING_2:
case ZONE_GALVANIZING_1:
case ZONE_GALVANIZING_2:
case ZONE_GALVANIZING_3:
case ZONE_GALVANIZING_4:
case ZONE_GALVANIZING_5:
case ZONE_GALVANIZING_6:
case ZONE_GALVANIZING_7:
case ZONE_GALVANIZING_8:
// время скапывания реактивов
// TODO добавить переменные времен скапывания
robot1_code.code[cmd_index++] = ROBOT_CMD_WAIT | 30;
break;
case ZONE_WASHING_1A:
case ZONE_WASHING_2A:
case ZONE_WASHING_3A:
case ZONE_WASHING_4A:
// время скапывания 1-го каскада промывок
robot1_code.code[cmd_index++] = ROBOT_CMD_WAIT | 3;
break;
case ZONE_WASHING_1B:
case ZONE_WASHING_2B:
case ZONE_WASHING_3B:
case ZONE_WASHING_4B:
// время скапывания 2-го каскада промывок
robot1_code.code[cmd_index++] = ROBOT_CMD_WAIT | 20;
break;
}
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_TO_ZONE_WITH_BARREL | tasks[target_task].dest_zone;
if (tasks[target_task].dest_zone == 22) {
robot1_code.code[cmd_index++] = ROBOT_CMD_DOWN_2_WITH_BARREL;
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_TO_ZONE | 21;
// NOTE старая механика не позволяет просто опустить траверсу до конца, для новой изменить поведение
robot1_code.code[cmd_index++] = ROBOT_CMD_UP;
robot1_code.code[cmd_index++] = ROBOT_CMD_DOWN;
} else {
robot1_code.code[cmd_index++] = ROBOT_CMD_DOWN_WITH_BARREL;
if (tasks[target_task].dest_zone != 0) {
// установка времени ожидания барабана
short tmp = -1;
switch (tasks[target_task].dest_zone) {
case ZONE_DEGREASING:
tmp = barrels[target_task].time_degreasing; break;
case ZONE_ETCHING_1:
case ZONE_ETCHING_2:
tmp = barrels[target_task].time_etching; break;
case ZONE_GALVANIZING_1:
case ZONE_GALVANIZING_2:
case ZONE_GALVANIZING_3:
case ZONE_GALVANIZING_4:
case ZONE_GALVANIZING_5:
case ZONE_GALVANIZING_6:
case ZONE_GALVANIZING_7:
case ZONE_GALVANIZING_8:
tmp = barrels[target_task].time_galvanizing; break;
case ZONE_WASHING_1A:
tmp = barrels[target_task].time_washing_1a; break;
case ZONE_WASHING_2A:
tmp = barrels[target_task].time_washing_2a; break;
case ZONE_WASHING_3A:
tmp = barrels[target_task].time_washing_3a; break;
case ZONE_WASHING_4A:
tmp = barrels[target_task].time_washing_4a; break;
case ZONE_WASHING_1B:
tmp = barrels[target_task].time_washing_1b; break;
case ZONE_WASHING_2B:
tmp = barrels[target_task].time_washing_2b; break;
case ZONE_WASHING_3B:
tmp = barrels[target_task].time_washing_3b; break;
case ZONE_WASHING_4B:
tmp = barrels[target_task].time_washing_4b; break;
}
if (tmp != -1) {
robot1_code.code[cmd_index++] = ROBOT_CMD_TMR_SET;
robot1_code.code[cmd_index++] = tmp;
}
}
}
} else {
// пассивация, тут все просто
if (robot1.dx.current_zone != tasks[target_task].start_zone) {
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_TO_ZONE | ZONE_WASHING_3B;
}
robot1_code.code[cmd_index++] = ROBOT_CMD_UP_WITH_BARREL;
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_TO_ZONE_WITH_BARREL | ZONE_PASSIVATION;
robot1_code.code[cmd_index++] = ROBOT_CMD_DOWN_WITH_BARREL;
robot1_code.code[cmd_index++] = ROBOT_CMD_TMR_SET;
robot1_code.code[cmd_index++] = barrels[target_task].time_passivation;
robot1_code.code[cmd_index++] = ROBOT_CMD_UP_WITH_BARREL;
robot1_code.code[cmd_index++] = ROBOT_CMD_MOVE_TO_ZONE_WITH_BARREL | ZONE_WASHING_4A;
robot1_code.code[cmd_index++] = ROBOT_CMD_DOWN_WITH_BARREL;
robot1_code.code[cmd_index++] = ROBOT_CMD_TMR_SET;
robot1_code.code[cmd_index++] = barrels[target_task].time_washing_4a;
}
robot1_code.code[cmd_index++] = ROBOT_CMD_END;
robot1_code.PC = 0;
printf("INFO: code length is %d\n", cmd_index);
debug_print_robot1_code();
}
} else {
printf("WARMING: support only one robot mode\n");
} }
} }
@ -397,7 +504,7 @@ void schedule_robot_1() {
char create_barrel_in_load(short zone) { char create_barrel_in_load(short zone) {
for (int i = 0; i < BARRELS_COUNT; i++) { for (int i = 0; i < BARRELS_COUNT; i++) {
if (!barrels[i].flags.is_exist) { if (!barrels[i].flags.is_exist) {
barrels[i] = makeBarrel(1, zone, 0, PROCESS_NONE); barrels[i] = makeBarrel(1, zone, 0);
return 0; return 0;
} }
} }
@ -419,30 +526,30 @@ char create_barrel_in_load(short zone) {
* съебать в 22 зону * съебать в 22 зону
иначе иначе
* съебать в зону изъятия * съебать в зону изъятия
* поднять траверсу * поднять траверсу с барабаном
если зона изъятия != 22 и зона изъятия != 0: если зона изъятия != 22 и зона изъятия != 1:
* ждать скапывания (зависит от зоны) * ждать скапывания (зависит от зоны)
* ехать в зону назначения * ехать в зону назначения
если зона назначения == 22 если зона назначения == 22
* опустить траверсу не до конца * опустить траверсу не до конца с барабаном
* съебать в 21 зону * съебать в 21 зону
если ЭНКОДЕРЫ СТАРЫЕ (по умолчанию) если ЭНКОДЕРЫ СТАРЫЕ (по умолчанию)
* поднять траверсу * поднять траверсу
* опустить траверсу * опустить траверсу
иначе иначе
* опустить траверсу * опустить траверсу с барабаном
если зона назначения != 1 если зона назначения != 0
* установить время ожидания барабана (зависит от зоны) * установить время ожидания барабана (зависит от зоны)
иначе иначе
если текущая зона != промывка 3б если текущая зона != промывка 3б
* съебать в промывку 3б * съебать в промывку 3б
* поднять траверсу * поднять траверсу с барабаном
* съебать в пассивацию * съебать в пассивацию
* опустить траверсу * опустить траверсу с барабаном
* поставить время ожидания барабана в <время пассивации> * поставить время ожидания барабана в <время пассивации>
* поднять траверсу * поднять траверсу с барабаном
* съебать в зону промывка 4а * съебать в зону промывка 4а
* опустить траверсу * опустить траверсу с барабаном
* установить время ожидания барабана (для промывки 4а) * установить время ожидания барабана (для промывки 4а)
*/ */
@ -465,7 +572,7 @@ void scheduler_main() {
} }
} }
} }
if (robot1_cmd.cmd == 0) { if (robot1_code.PC < 0) {
schedule_robot_1(); schedule_robot_1();
} }

View File

@ -2,5 +2,3 @@
struct barrel barrels[BARRELS_COUNT]; struct barrel barrels[BARRELS_COUNT];
struct robot_cmd robot1_cmd;
struct robot_cmd robot2_cmd;

64
utils.h
View File

@ -49,10 +49,9 @@ struct barrel {
short zone; short zone;
// программный таймер, тикает сразу у всех барабанов, // программный таймер, тикает сразу у всех барабанов,
short software_timer; short software_timer;
short curr_process; // стадия процесса
// время процессов // время процессов
short time_defatting; // Время обезжиривания short time_degreasing; // Время обезжиривания
short time_washing_1a; // Время промывки 1А short time_washing_1a; // Время промывки 1А
short time_washing_1b; // Время промывки 1Б short time_washing_1b; // Время промывки 1Б
short time_etching; // Время травления short time_etching; // Время травления
@ -66,30 +65,6 @@ struct barrel {
short time_washing_4b; // Время промывки 4Б short time_washing_4b; // Время промывки 4Б
}; };
// deprecated: do not use!
enum BarrelProcess {
PROCESS_NONE = 0, // сразу после загрузки
PROCESS_DEFATTING, // обезжиривание
PROCESS_WASHING_1A, // промывка 1А
PROCESS_WASHING_1B, // промывка 1Б
PROCESS_ETCHING, // травление
PROCESS_WASHING_2A, // промывка 2А
PROCESS_WASHING_2B, // промывка 2Б
PROCESS_GALVANIZING, // цинкование
PROCESS_WASHING_3A, // промывка 3А
PROCESS_WASHING_3B, // промывка 3Б
PROCESS_PASSIVATION, // пассивация
PROCESS_WASHING_4B, // промывка 4Б
PROCESS_RETURN_1, // возвращение обратно, стадия 1 - перекладываем в свободную промывку (если есть)
PROCESS_RETURN_2, // возвращение обратно, стадия 2 - перекладываем в свободную загрузку и удалить барабан
};
struct robot_cmd {
short cmd;
short args[2];
short step;
};
union robot_regs_mx { union robot_regs_mx {
short raw_word; short raw_word;
@ -159,8 +134,6 @@ extern "C" {
extern struct barrel barrels[BARRELS_COUNT]; extern struct barrel barrels[BARRELS_COUNT];
extern struct robot_cmd robot1_cmd;
extern struct robot_cmd robot2_cmd;
// почему-то компилер говорит что размер структуру сраный // почему-то компилер говорит что размер структуру сраный
//extern struct robot robot1; //extern struct robot robot1;
@ -176,9 +149,28 @@ extern struct robot_cmd robot2_cmd;
#define ROBOT_CMD_WAIT 0x6000 #define ROBOT_CMD_WAIT 0x6000
#define ROBOT_CMD_TMR_SET 0x7000 #define ROBOT_CMD_TMR_SET 0x7000
#define ROBOT_CMD_SET_LOCK_ZONE 0x8000 #define ROBOT_CMD_SET_LOCK_ZONE 0x8000
#define ROBOT_CMD_CORRECT_X 0x9000 #define ROBOT_CMD_CORRECT_AXIS 0x9000
#define ROBOT_CMD_CORRECT_Z 0xA000 #define ROBOT_CMD_INC_ZONE 0xA000
// перемещение с барабаном
#define ROBOT_WITH_BARREL 0x0800
// алиасы перемещений с барабаном
#define ROBOT_CMD_MOVE_TO_ZONE_WITH_BARREL (ROBOT_CMD_MOVE_TO_ZONE | ROBOT_WITH_BARREL)
#define ROBOT_CMD_UP_WITH_BARREL (ROBOT_CMD_UP | ROBOT_WITH_BARREL)
#define ROBOT_CMD_DOWN_WITH_BARREL (ROBOT_CMD_DOWN | ROBOT_WITH_BARREL)
#define ROBOT_CMD_DOWN_2_WITH_BARREL (ROBOT_CMD_DOWN_2 | ROBOT_WITH_BARREL)
#define ROBOT_AXIS_X 1
#define ROBOT_CMD_CORRECT_AXIS_X (ROBOT_CMD_CORRECT_AXIS | ROBOT_AXIS_X)
#define ROBOT_AXIS_Z 2
#define ROBOT_CMD_CORRECT_AXIS_Z (ROBOT_CMD_CORRECT_AXIS | ROBOT_AXIS_Z)
#define ROBOT_ZONE_ETCH 1
#define ROBOT_CMD_INC_ZONE_ETCH (ROBOT_CMD_INC_ZONE | ROBOT_ZONE_ETCH)
#define ROBOT_ZONE_GAL 2
#define ROBOT_CMD_INC_ZONE_GAL (ROBOT_CMD_INC_ZONE | ROBOT_ZONE_GAL)
// NOTE первой командой на любую транзакцию должна стоять команда опустить траверсу (в 22 зоне по идее никогда не закончим) // NOTE первой командой на любую транзакцию должна стоять команда опустить траверсу (в 22 зоне по идее никогда не закончим)
struct robot_code { struct robot_code {
@ -188,16 +180,16 @@ struct robot_code {
/* /*
* система команд, которая нужна: (квадратные скобки - аргумент это младший байт, фигурные - отдельное слово) * система команд, которая нужна: (квадратные скобки - аргумент это младший байт, фигурные - отдельное слово)
* 0: конец * 0: конец
* 1 [зона]: съебаться в зону * 1 (опция с барабаном) [зона]: съебаться в зону
* 2: встать на смещенную позицию * 2: встать на смещенную позицию
* 3 [с барабаном]: поднять траверсу (перед выполнением ожидать если таймер барабана не истек) * 3 (опция с барабаном): поднять траверсу (перед выполнением ожидать если таймер барабана не истек)
* 4 [с барабаном]: опустить траверсу * 4 (опция с барабаном): опустить траверсу
* 5 [с барабаном]: опустить траверсу не до конца * 5 (опция с барабаном): опустить траверсу не до конца
* 6 [сек]: пауза на нужное количество секунд * 6 [сек]: пауза на нужное количество секунд
* 7 {время}: установить таймер барабану * 7 {время}: установить таймер барабану
* 8 [зона]: установить зону блокировки * 8 [зона]: установить зону блокировки
* 9: скорректировать ос X * 9 [X(1) | Y(2)]: скорректировать ось
* 10: скорректировать ос Z * 10 [ETCH(1) |GAL (2)]: сделать инкремент зоны гальваники или обезжиривания
* *
* формат команды: (команда, старший байт) [младший байт, аргумент команды (если есть)] [слово, аргумент если команда требует] * формат команды: (команда, старший байт) [младший байт, аргумент команды (если есть)] [слово, аргумент если команда требует]
*/ */