2016-05-25 18:37:40 +03:00
|
|
|
# Wrapper around C helper code
|
|
|
|
#
|
2017-04-13 19:09:37 +03:00
|
|
|
# Copyright (C) 2016,2017 Kevin O'Connor <kevin@koconnor.net>
|
2016-05-25 18:37:40 +03:00
|
|
|
#
|
|
|
|
# This file may be distributed under the terms of the GNU GPLv3 license.
|
|
|
|
import os, logging
|
|
|
|
import cffi
|
|
|
|
|
2017-04-13 19:09:37 +03:00
|
|
|
|
|
|
|
######################################################################
|
|
|
|
# c_helper.so compiling
|
|
|
|
######################################################################
|
|
|
|
|
2018-06-06 04:40:32 +03:00
|
|
|
COMPILE_CMD = ("gcc -Wall -g -O2 -shared -fPIC"
|
|
|
|
" -flto -fwhole-program -fno-use-linker-plugin"
|
|
|
|
" -o %s %s")
|
2018-06-06 23:49:44 +03:00
|
|
|
SOURCE_FILES = [
|
2018-06-09 21:30:22 +03:00
|
|
|
'pyhelper.c', 'serialqueue.c', 'stepcompress.c', 'itersolve.c',
|
2018-06-13 23:53:25 +03:00
|
|
|
'kin_cartesian.c', 'kin_corexy.c', 'kin_delta.c', 'kin_extruder.c'
|
2018-06-06 23:49:44 +03:00
|
|
|
]
|
2016-05-25 18:37:40 +03:00
|
|
|
DEST_LIB = "c_helper.so"
|
2018-06-09 02:55:24 +03:00
|
|
|
OTHER_FILES = [
|
|
|
|
'list.h', 'serialqueue.h', 'stepcompress.h', 'itersolve.h', 'pyhelper.h'
|
|
|
|
]
|
2016-05-25 18:37:40 +03:00
|
|
|
|
|
|
|
defs_stepcompress = """
|
2018-06-09 00:02:20 +03:00
|
|
|
struct stepcompress *stepcompress_alloc(uint32_t oid);
|
|
|
|
void stepcompress_fill(struct stepcompress *sc, uint32_t max_error
|
|
|
|
, uint32_t invert_sdir, uint32_t queue_step_msgid
|
|
|
|
, uint32_t set_next_step_dir_msgid);
|
2016-11-30 09:58:45 +03:00
|
|
|
void stepcompress_free(struct stepcompress *sc);
|
2017-04-05 02:37:54 +03:00
|
|
|
int stepcompress_reset(struct stepcompress *sc, uint64_t last_step_clock);
|
|
|
|
int stepcompress_set_homing(struct stepcompress *sc, uint64_t homing_clock);
|
|
|
|
int stepcompress_queue_msg(struct stepcompress *sc, uint32_t *data, int len);
|
|
|
|
|
2018-06-06 23:49:44 +03:00
|
|
|
struct steppersync *steppersync_alloc(struct serialqueue *sq
|
|
|
|
, struct stepcompress **sc_list, int sc_num, int move_num);
|
|
|
|
void steppersync_free(struct steppersync *ss);
|
|
|
|
void steppersync_set_time(struct steppersync *ss
|
|
|
|
, double time_offset, double mcu_freq);
|
|
|
|
int steppersync_flush(struct steppersync *ss, uint64_t move_clock);
|
|
|
|
"""
|
|
|
|
|
2018-06-09 02:55:24 +03:00
|
|
|
defs_itersolve = """
|
|
|
|
struct move *move_alloc(void);
|
|
|
|
void move_fill(struct move *m, double print_time
|
|
|
|
, double accel_t, double cruise_t, double decel_t
|
|
|
|
, double start_pos_x, double start_pos_y, double start_pos_z
|
|
|
|
, double axes_d_x, double axes_d_y, double axes_d_z
|
|
|
|
, double start_v, double cruise_v, double accel);
|
|
|
|
int32_t itersolve_gen_steps(struct stepper_kinematics *sk, struct move *m);
|
|
|
|
void itersolve_set_stepcompress(struct stepper_kinematics *sk
|
|
|
|
, struct stepcompress *sc, double step_dist);
|
2018-06-22 20:03:07 +03:00
|
|
|
void itersolve_set_position(struct stepper_kinematics *sk
|
|
|
|
, double x, double y, double z);
|
2018-06-14 23:33:57 +03:00
|
|
|
void itersolve_set_commanded_pos(struct stepper_kinematics *sk, double pos);
|
|
|
|
double itersolve_get_commanded_pos(struct stepper_kinematics *sk);
|
2018-06-09 02:55:24 +03:00
|
|
|
"""
|
|
|
|
|
2018-06-06 23:49:44 +03:00
|
|
|
defs_kin_cartesian = """
|
2018-06-09 21:30:22 +03:00
|
|
|
struct stepper_kinematics *cartesian_stepper_alloc(char axis);
|
|
|
|
"""
|
|
|
|
|
|
|
|
defs_kin_corexy = """
|
|
|
|
struct stepper_kinematics *corexy_stepper_alloc(char type);
|
2018-06-06 23:49:44 +03:00
|
|
|
"""
|
|
|
|
|
|
|
|
defs_kin_delta = """
|
2018-06-09 04:30:38 +03:00
|
|
|
struct stepper_kinematics *delta_stepper_alloc(double arm2
|
|
|
|
, double tower_x, double tower_y);
|
2016-05-25 18:37:40 +03:00
|
|
|
"""
|
|
|
|
|
2018-06-13 23:53:25 +03:00
|
|
|
defs_kin_extruder = """
|
|
|
|
struct stepper_kinematics *extruder_stepper_alloc(void);
|
|
|
|
void extruder_move_fill(struct move *m, double print_time
|
|
|
|
, double accel_t, double cruise_t, double decel_t, double start_pos
|
|
|
|
, double start_v, double cruise_v, double accel
|
|
|
|
, double extra_accel_v, double extra_decel_v);
|
|
|
|
"""
|
|
|
|
|
2016-05-25 18:37:40 +03:00
|
|
|
defs_serialqueue = """
|
|
|
|
#define MESSAGE_MAX 64
|
|
|
|
struct pull_queue_message {
|
|
|
|
uint8_t msg[MESSAGE_MAX];
|
|
|
|
int len;
|
|
|
|
double sent_time, receive_time;
|
|
|
|
};
|
|
|
|
|
2016-07-11 18:41:49 +03:00
|
|
|
struct serialqueue *serialqueue_alloc(int serial_fd, int write_only);
|
2016-05-25 18:37:40 +03:00
|
|
|
void serialqueue_exit(struct serialqueue *sq);
|
2016-11-30 00:22:54 +03:00
|
|
|
void serialqueue_free(struct serialqueue *sq);
|
2016-05-25 18:37:40 +03:00
|
|
|
struct command_queue *serialqueue_alloc_commandqueue(void);
|
2016-11-30 09:58:45 +03:00
|
|
|
void serialqueue_free_commandqueue(struct command_queue *cq);
|
2016-05-25 18:37:40 +03:00
|
|
|
void serialqueue_send(struct serialqueue *sq, struct command_queue *cq
|
|
|
|
, uint8_t *msg, int len, uint64_t min_clock, uint64_t req_clock);
|
2018-05-28 16:42:59 +03:00
|
|
|
void serialqueue_pull(struct serialqueue *sq
|
|
|
|
, struct pull_queue_message *pqm);
|
2016-07-11 18:41:49 +03:00
|
|
|
void serialqueue_set_baud_adjust(struct serialqueue *sq, double baud_adjust);
|
2018-05-28 16:42:59 +03:00
|
|
|
void serialqueue_set_receive_window(struct serialqueue *sq
|
|
|
|
, int receive_window);
|
2017-06-28 03:08:18 +03:00
|
|
|
void serialqueue_set_clock_est(struct serialqueue *sq, double est_freq
|
|
|
|
, double last_clock_time, uint64_t last_clock);
|
2016-05-25 18:37:40 +03:00
|
|
|
void serialqueue_get_stats(struct serialqueue *sq, char *buf, int len);
|
|
|
|
int serialqueue_extract_old(struct serialqueue *sq, int sentq
|
|
|
|
, struct pull_queue_message *q, int max);
|
|
|
|
"""
|
|
|
|
|
2016-11-30 20:04:28 +03:00
|
|
|
defs_pyhelper = """
|
|
|
|
void set_python_logging_callback(void (*func)(const char *));
|
2017-02-06 21:31:34 +03:00
|
|
|
double get_monotonic(void);
|
2016-11-30 20:04:28 +03:00
|
|
|
"""
|
|
|
|
|
2018-06-09 02:55:24 +03:00
|
|
|
defs_std = """
|
|
|
|
void free(void*);
|
|
|
|
"""
|
|
|
|
|
2018-06-06 23:49:44 +03:00
|
|
|
defs_all = [
|
2018-06-09 21:30:22 +03:00
|
|
|
defs_pyhelper, defs_serialqueue, defs_std, defs_stepcompress, defs_itersolve,
|
2018-06-13 23:53:25 +03:00
|
|
|
defs_kin_cartesian, defs_kin_corexy, defs_kin_delta, defs_kin_extruder
|
2018-06-06 23:49:44 +03:00
|
|
|
]
|
|
|
|
|
2016-05-25 18:37:40 +03:00
|
|
|
# Return the list of file modification times
|
|
|
|
def get_mtimes(srcdir, filelist):
|
|
|
|
out = []
|
|
|
|
for filename in filelist:
|
|
|
|
pathname = os.path.join(srcdir, filename)
|
|
|
|
try:
|
|
|
|
t = os.path.getmtime(pathname)
|
|
|
|
except os.error:
|
|
|
|
continue
|
|
|
|
out.append(t)
|
|
|
|
return out
|
|
|
|
|
|
|
|
# Check if the code needs to be compiled
|
2017-04-13 19:09:37 +03:00
|
|
|
def check_build_code(srcdir, target, sources, cmd, other_files=[]):
|
|
|
|
src_times = get_mtimes(srcdir, sources + other_files)
|
|
|
|
obj_times = get_mtimes(srcdir, [target])
|
2016-05-25 18:37:40 +03:00
|
|
|
if not obj_times or max(src_times) > min(obj_times):
|
2017-09-27 18:43:14 +03:00
|
|
|
logging.info("Building C code module %s", target)
|
2017-04-13 19:09:37 +03:00
|
|
|
srcfiles = [os.path.join(srcdir, fname) for fname in sources]
|
|
|
|
destlib = os.path.join(srcdir, target)
|
|
|
|
os.system(cmd % (destlib, ' '.join(srcfiles)))
|
2016-05-25 18:37:40 +03:00
|
|
|
|
|
|
|
FFI_main = None
|
|
|
|
FFI_lib = None
|
2016-11-30 20:04:28 +03:00
|
|
|
pyhelper_logging_callback = None
|
2016-05-25 18:37:40 +03:00
|
|
|
|
|
|
|
# Return the Foreign Function Interface api to the caller
|
|
|
|
def get_ffi():
|
2016-11-30 20:04:28 +03:00
|
|
|
global FFI_main, FFI_lib, pyhelper_logging_callback
|
2016-05-25 18:37:40 +03:00
|
|
|
if FFI_lib is None:
|
|
|
|
srcdir = os.path.dirname(os.path.realpath(__file__))
|
2017-04-13 19:09:37 +03:00
|
|
|
check_build_code(srcdir, DEST_LIB, SOURCE_FILES, COMPILE_CMD
|
|
|
|
, OTHER_FILES)
|
2016-05-25 18:37:40 +03:00
|
|
|
FFI_main = cffi.FFI()
|
2018-06-06 23:49:44 +03:00
|
|
|
for d in defs_all:
|
|
|
|
FFI_main.cdef(d)
|
2016-05-25 18:37:40 +03:00
|
|
|
FFI_lib = FFI_main.dlopen(os.path.join(srcdir, DEST_LIB))
|
2016-11-30 20:04:28 +03:00
|
|
|
# Setup error logging
|
|
|
|
def logging_callback(msg):
|
|
|
|
logging.error(FFI_main.string(msg))
|
|
|
|
pyhelper_logging_callback = FFI_main.callback(
|
|
|
|
"void(const char *)", logging_callback)
|
|
|
|
FFI_lib.set_python_logging_callback(pyhelper_logging_callback)
|
2016-05-25 18:37:40 +03:00
|
|
|
return FFI_main, FFI_lib
|
2017-04-13 19:09:37 +03:00
|
|
|
|
|
|
|
|
|
|
|
######################################################################
|
|
|
|
# hub-ctrl hub power controller
|
|
|
|
######################################################################
|
|
|
|
|
|
|
|
HC_COMPILE_CMD = "gcc -Wall -g -O2 -o %s %s -lusb"
|
|
|
|
HC_SOURCE_FILES = ['hub-ctrl.c']
|
2018-05-05 16:54:54 +03:00
|
|
|
HC_SOURCE_DIR = '../../lib/hub-ctrl'
|
2017-04-13 19:09:37 +03:00
|
|
|
HC_TARGET = "hub-ctrl"
|
|
|
|
HC_CMD = "sudo %s/hub-ctrl -h 0 -P 2 -p %d"
|
|
|
|
|
|
|
|
def run_hub_ctrl(enable_power):
|
|
|
|
srcdir = os.path.dirname(os.path.realpath(__file__))
|
|
|
|
hubdir = os.path.join(srcdir, HC_SOURCE_DIR)
|
|
|
|
check_build_code(hubdir, HC_TARGET, HC_SOURCE_FILES, HC_COMPILE_CMD)
|
|
|
|
os.system(HC_CMD % (hubdir, enable_power))
|