2011-07-09 12:37:34 +00:00
|
|
|
/*
|
|
|
|
* Hamlib ts7400 backend - main file
|
|
|
|
* Copyright (c) 2001-2009 by Stephane Fillod
|
|
|
|
*
|
|
|
|
*
|
2011-08-22 02:01:25 +00:00
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
2011-07-09 12:37:34 +00:00
|
|
|
*
|
2011-08-22 02:01:25 +00:00
|
|
|
* This library is distributed in the hope that it will be useful,
|
2011-07-09 12:37:34 +00:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2011-08-22 02:01:25 +00:00
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
2011-07-09 12:37:34 +00:00
|
|
|
*
|
2011-08-22 02:01:25 +00:00
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2011-07-09 12:37:34 +00:00
|
|
|
* License along with this library; if not, write to the Free Software
|
2011-08-22 02:01:25 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
2011-07-09 12:37:34 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h> /* String function definitions */
|
|
|
|
#include <unistd.h> /* UNIX standard function definitions */
|
|
|
|
#include <math.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
#include <hamlib/rotator.h>
|
|
|
|
#include "serial.h"
|
|
|
|
#include "misc.h"
|
|
|
|
#include "register.h"
|
|
|
|
|
|
|
|
#include "ts7400.h"
|
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
struct ts7400_rot_priv_data
|
|
|
|
{
|
|
|
|
azimuth_t az;
|
|
|
|
elevation_t el;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
struct timeval tv; /* time last az/el update */
|
|
|
|
azimuth_t target_az;
|
|
|
|
elevation_t target_el;
|
2011-07-09 12:37:34 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int ts7400_rot_init(ROT *rot)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
struct ts7400_rot_priv_data *priv;
|
|
|
|
|
2020-01-13 18:13:36 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
|
|
|
|
|
|
|
rot->state.priv = (struct ts7400_rot_priv_data *)
|
2020-02-23 17:26:09 +00:00
|
|
|
malloc(sizeof(struct ts7400_rot_priv_data));
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2020-01-13 18:13:36 +00:00
|
|
|
if (!rot->state.priv)
|
2019-11-30 16:19:08 +00:00
|
|
|
{
|
|
|
|
return -RIG_ENOMEM;
|
|
|
|
}
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2020-01-13 18:13:36 +00:00
|
|
|
priv = rot->state.priv;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
rot->state.rotport.type.rig = RIG_PORT_NONE;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
priv->az = priv->el = 0;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
priv->target_az = priv->target_el = 0;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int ts7400_rot_cleanup(ROT *rot)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
if (rot->state.priv)
|
|
|
|
{
|
|
|
|
free(rot->state.priv);
|
|
|
|
}
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
rot->state.priv = NULL;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int ts7400_rot_open(ROT *rot)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int ts7400_rot_close(ROT *rot)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int ts7400_rot_set_position(ROT *rot, azimuth_t az, elevation_t el)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
struct ts7400_rot_priv_data *priv = (struct ts7400_rot_priv_data *)
|
|
|
|
rot->state.priv;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called: %.2f %.2f\n", __func__,
|
|
|
|
az, el);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
priv->target_az = az;
|
|
|
|
priv->target_el = el;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
gettimeofday(&priv->tv, NULL);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get position of rotor, simulating slow rotation
|
|
|
|
*/
|
|
|
|
static int ts7400_rot_get_position(ROT *rot, azimuth_t *az, elevation_t *el)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
struct ts7400_rot_priv_data *priv = (struct ts7400_rot_priv_data *)
|
|
|
|
rot->state.priv;
|
|
|
|
struct timeval tv;
|
|
|
|
unsigned elapsed; /* ms */
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
if (priv->az == priv->target_az &&
|
|
|
|
priv->el == priv->target_el)
|
|
|
|
{
|
|
|
|
*az = priv->az;
|
|
|
|
*el = priv->el;
|
|
|
|
return RIG_OK;
|
|
|
|
}
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
gettimeofday(&tv, NULL);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
elapsed = (tv.tv_sec - priv->tv.tv_sec) * 1000 +
|
|
|
|
(tv.tv_usec - priv->tv.tv_usec) / 1000;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
/*
|
|
|
|
* Simulate rotation speed of 360 deg per minute
|
|
|
|
*/
|
2011-07-09 12:37:34 +00:00
|
|
|
#define DEG_PER_MS (360./60/1000)
|
2019-11-30 16:19:08 +00:00
|
|
|
|
|
|
|
if (fabs(priv->target_az - priv->az) / DEG_PER_MS <= elapsed)
|
|
|
|
{
|
|
|
|
/* target reached */
|
|
|
|
priv->az = priv->target_az;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (priv->az < priv->target_az)
|
|
|
|
{
|
|
|
|
priv->az += (azimuth_t)elapsed * DEG_PER_MS;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
priv->az -= (azimuth_t)elapsed * DEG_PER_MS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fabs(priv->target_el - priv->el) / DEG_PER_MS <= elapsed)
|
|
|
|
{
|
|
|
|
/* target reached */
|
|
|
|
priv->el = priv->target_el;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (priv->el < priv->target_el)
|
|
|
|
{
|
|
|
|
priv->el += (elevation_t)elapsed * DEG_PER_MS;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
priv->el -= (elevation_t)elapsed * DEG_PER_MS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*az = priv->az;
|
|
|
|
*el = priv->el;
|
|
|
|
|
|
|
|
priv->tv = tv;
|
|
|
|
|
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int ts7400_rot_stop(ROT *rot)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
struct ts7400_rot_priv_data *priv = (struct ts7400_rot_priv_data *)
|
|
|
|
rot->state.priv;
|
|
|
|
azimuth_t az;
|
|
|
|
elevation_t el;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
ts7400_rot_get_position(rot, &az, &el);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
priv->target_az = priv->az = az;
|
|
|
|
priv->target_el = priv->el = el;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int ts7400_rot_park(ROT *rot)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
/* Assume home is 0,0 */
|
|
|
|
ts7400_rot_set_position(rot, 0, 0);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int ts7400_rot_reset(ROT *rot, rot_reset_t reset)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int ts7400_rot_move(ROT *rot, int direction, int speed)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
struct ts7400_rot_priv_data *priv = (struct ts7400_rot_priv_data *)
|
|
|
|
rot->state.priv;
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
|
|
|
rig_debug(RIG_DEBUG_TRACE, "%s: Direction = %d, Speed = %d\n", __func__,
|
|
|
|
direction, speed);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
switch (direction)
|
|
|
|
{
|
|
|
|
case ROT_MOVE_UP:
|
|
|
|
return ts7400_rot_set_position(rot, priv->target_az, 90);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
case ROT_MOVE_DOWN:
|
|
|
|
return ts7400_rot_set_position(rot, priv->target_az, 0);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
case ROT_MOVE_CCW:
|
|
|
|
return ts7400_rot_set_position(rot, -180, priv->target_el);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
case ROT_MOVE_CW:
|
|
|
|
return ts7400_rot_set_position(rot, 180, priv->target_el);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
default:
|
|
|
|
return -RIG_EINVAL;
|
|
|
|
}
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const char *ts7400_rot_get_info(ROT *rot)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s called\n", __func__);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return "ts7400 rotator";
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ts7400 rotator capabilities.
|
|
|
|
*/
|
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
const struct rot_caps ts7400_rot_caps =
|
|
|
|
{
|
2020-03-06 05:48:14 +00:00
|
|
|
ROT_MODEL(ROT_MODEL_TS7400),
|
2019-11-30 16:19:08 +00:00
|
|
|
.model_name = "ts7400",
|
|
|
|
.mfg_name = "LA7LKA",
|
2020-06-08 21:13:08 +00:00
|
|
|
.version = "20200113.0",
|
2019-11-30 16:19:08 +00:00
|
|
|
.copyright = "LGPL",
|
|
|
|
.status = RIG_STATUS_BETA,
|
|
|
|
.rot_type = ROT_TYPE_AZEL,
|
|
|
|
.port_type = RIG_PORT_NONE,
|
|
|
|
|
|
|
|
.min_az = -180.,
|
|
|
|
.max_az = 180.,
|
|
|
|
.min_el = 0.,
|
|
|
|
.max_el = 90.,
|
|
|
|
|
|
|
|
.priv = NULL, /* priv */
|
|
|
|
|
|
|
|
.rot_init = ts7400_rot_init,
|
|
|
|
.rot_cleanup = ts7400_rot_cleanup,
|
|
|
|
.rot_open = ts7400_rot_open,
|
|
|
|
.rot_close = ts7400_rot_close,
|
|
|
|
|
|
|
|
.set_position = ts7400_rot_set_position,
|
|
|
|
.get_position = ts7400_rot_get_position,
|
|
|
|
.park = ts7400_rot_park,
|
|
|
|
.stop = ts7400_rot_stop,
|
|
|
|
.reset = ts7400_rot_reset,
|
|
|
|
.move = ts7400_rot_move,
|
|
|
|
|
|
|
|
.get_info = ts7400_rot_get_info,
|
2011-07-09 12:37:34 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
DECLARE_INITROT_BACKEND(ts7400)
|
|
|
|
{
|
2019-11-30 16:19:08 +00:00
|
|
|
rig_debug(RIG_DEBUG_VERBOSE, "%s: _init called\n", __func__);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
rot_register(&ts7400_rot_caps);
|
2011-07-09 12:37:34 +00:00
|
|
|
|
2019-11-30 16:19:08 +00:00
|
|
|
return RIG_OK;
|
2011-07-09 12:37:34 +00:00
|
|
|
}
|