Kstars

phd2.cpp
1/*
2 SPDX-FileCopyrightText: 2016 Jasem Mutlaq <mutlaqja@ikarustech.com>
3
4 SPDX-License-Identifier: GPL-2.0-or-later
5*/
6
7#include "phd2.h"
8
9#include "Options.h"
10#include "kspaths.h"
11#include "kstars.h"
12
13#include "ekos/manager.h"
14#include "fitsviewer/fitsdata.h"
15#include "ekos/guide/guide.h"
16#include "fitsviewer/fitsview.h"
17
18#include <cassert>
19#include <fitsio.h>
20#include <KMessageBox>
21#include <QImage>
22
23#include <QJsonDocument>
24#include <QNetworkReply>
25
26#include <ekos_guide_debug.h>
27
28#define MAX_SET_CONNECTED_RETRIES 3
29
30namespace Ekos
31{
32PHD2::PHD2()
33{
34 tcpSocket = new QTcpSocket(this);
35
36 //This list of available PHD Events is on https://github.com/OpenPHDGuiding/phd2/wiki/EventMonitoring
37
38 events["Version"] = Version;
39 events["LockPositionSet"] = LockPositionSet;
40 events["Calibrating"] = Calibrating;
41 events["CalibrationComplete"] = CalibrationComplete;
42 events["StarSelected"] = StarSelected;
43 events["StartGuiding"] = StartGuiding;
44 events["Paused"] = Paused;
45 events["StartCalibration"] = StartCalibration;
46 events["AppState"] = AppState;
47 events["CalibrationFailed"] = CalibrationFailed;
48 events["CalibrationDataFlipped"] = CalibrationDataFlipped;
49 events["LoopingExposures"] = LoopingExposures;
50 events["LoopingExposuresStopped"] = LoopingExposuresStopped;
51 events["SettleBegin"] = SettleBegin;
52 events["Settling"] = Settling;
53 events["SettleDone"] = SettleDone;
54 events["StarLost"] = StarLost;
55 events["GuidingStopped"] = GuidingStopped;
56 events["Resumed"] = Resumed;
57 events["GuideStep"] = GuideStep;
58 events["GuidingDithered"] = GuidingDithered;
59 events["LockPositionLost"] = LockPositionLost;
60 events["Alert"] = Alert;
61 events["GuideParamChange"] = GuideParamChange;
62 events["ConfigurationChange"] = ConfigurationChange;
63
64 //This list of available PHD Methods is on https://github.com/OpenPHDGuiding/phd2/wiki/EventMonitoring
65 //Only some of the methods are implemented. The ones that say COMMAND_RECEIVED simply return a 0 saying the command was received.
66 methodResults["capture_single_frame"] = CAPTURE_SINGLE_FRAME;
67 methodResults["clear_calibration"] = CLEAR_CALIBRATION_COMMAND_RECEIVED;
68 methodResults["dither"] = DITHER_COMMAND_RECEIVED;
69 //find_star
70 //flip_calibration
71 //get_algo_param_names
72 //get_algo_param
73 methodResults["get_app_state"] = APP_STATE_RECEIVED;
74 //get_calibrated
75 //get_calibration_data
76 methodResults["get_connected"] = IS_EQUIPMENT_CONNECTED;
77 //get_cooler_status
78 methodResults["get_current_equipment"] = GET_CURRENT_EQUIPMENT;
79 methodResults["get_dec_guide_mode"] = DEC_GUIDE_MODE;
80 methodResults["get_exposure"] = EXPOSURE_TIME;
81 methodResults["get_exposure_durations"] = EXPOSURE_DURATIONS;
82 methodResults["get_lock_position"] = LOCK_POSITION;
83 //get_lock_shift_enabled
84 //get_lock_shift_params
85 //get_paused
86 methodResults["get_pixel_scale"] = PIXEL_SCALE;
87 //get_profile
88 //get_profiles
89 //get_search_region
90 //get_sensor_temperature
91 methodResults["get_star_image"] = STAR_IMAGE;
92 //get_use_subframes
93 methodResults["guide"] = GUIDE_COMMAND_RECEIVED;
94 //guide_pulse
95 methodResults["loop"] = LOOP;
96 //save_image
97 //set_algo_param
98 methodResults["set_connected"] = CONNECTION_RESULT;
99 methodResults["set_dec_guide_mode"] = SET_DEC_GUIDE_MODE_COMMAND_RECEIVED;
100 methodResults["set_exposure"] = SET_EXPOSURE_COMMAND_RECEIVED;
101 methodResults["set_lock_position"] = SET_LOCK_POSITION;
102 //set_lock_shift_enabled
103 //set_lock_shift_params
104 methodResults["set_paused"] = SET_PAUSED_COMMAND_RECEIVED;
105 //set_profile
106 //shutdown
107 methodResults["stop_capture"] = STOP_CAPTURE_COMMAND_RECEIVED;
108
109 abortTimer = new QTimer(this);
110 connect(abortTimer, &QTimer::timeout, this, [ = ]
111 {
112 if (state == CALIBRATING)
113 qCDebug(KSTARS_EKOS_GUIDE) << "Abort timeout expired while calibrating, retrying to guide.";
114 else if (state == LOSTLOCK)
115 qCDebug(KSTARS_EKOS_GUIDE) << "Abort timeout expired while reacquiring star, retrying to guide.";
116 else
117 qCDebug(KSTARS_EKOS_GUIDE) << "Abort timeout expired, stopping.";
118 abort();
119 });
120
121 ditherTimer = new QTimer(this);
122 connect(ditherTimer, &QTimer::timeout, this, [ = ]
123 {
124 qCDebug(KSTARS_EKOS_GUIDE) << "ditherTimer expired, state" << state << "dithering" << isDitherActive << "settling" << isSettling;
125 ditherTimer->stop();
126 isDitherActive = false;
127 isSettling = false;
128 if (Options::ditherFailAbortsAutoGuide())
129 {
130 abort();
131 emit newStatus(GUIDE_DITHERING_ERROR);
132 }
133 else
134 {
135 emit newLog(i18n("PHD2: There was no dithering response from PHD2, but continue guiding."));
136 emit newStatus(Ekos::GUIDE_DITHERING_SUCCESS);
137 }
138 });
139
140 stateTimer = new QTimer(this);
141 connect(stateTimer, &QTimer::timeout, this, [ = ]
142 {
143 QTcpSocket::SocketState socketstate = tcpSocket->state();
144 switch (socketstate)
145 {
147 m_PHD2ReconnectCounter++;
148 if (m_PHD2ReconnectCounter > PHD2_RECONNECT_THRESHOLD)
149 {
150 stateTimer->stop();
151 emit newLog(i18n("Giving up reconnecting."));
152 }
153 else
154 {
155 emit newLog(i18n("Reconnecting to PHD2 Host: %1, on port %2. . .", Options::pHD2Host(), Options::pHD2Port()));
156
157 connect(tcpSocket, &QTcpSocket::readyRead, this, &PHD2::readPHD2, Qt::UniqueConnection);
158#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
159 connect(tcpSocket, &QTcpSocket::errorOccurred, this, &PHD2::displayError, Qt::UniqueConnection);
160#else
161 connect(tcpSocket, SIGNAL(error(QAbstractSocket::SocketError)), this,
162 SLOT(displayError(QAbstractSocket::SocketError)));
163#endif
164 tcpSocket->connectToHost(Options::pHD2Host(), Options::pHD2Port());
165 }
166 break;
168 m_PHD2ReconnectCounter = 0;
169 checkIfEquipmentConnected();
170 requestAppState();
171 break;
172 default:
173 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: TCP connection state:" << socketstate;
174 break;
175 }
176 });
177}
178
179PHD2::~PHD2()
180{
181 delete abortTimer;
182 delete ditherTimer;
183}
184
185bool PHD2::Connect()
186{
187 switch (connection)
188 {
189 case DISCONNECTED:
190 // Not yet connected, let's connect server
191 connection = CONNECTING;
192 emit newLog(i18n("Connecting to PHD2 Host: %1, on port %2. . .", Options::pHD2Host(), Options::pHD2Port()));
193
194 connect(tcpSocket, &QTcpSocket::readyRead, this, &PHD2::readPHD2, Qt::UniqueConnection);
195#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
196 connect(tcpSocket, &QTcpSocket::errorOccurred, this, &PHD2::displayError, Qt::UniqueConnection);
197#else
198 connect(tcpSocket, SIGNAL(error(QAbstractSocket::SocketError)), this,
199 SLOT(displayError(QAbstractSocket::SocketError)));
200#endif
201
202 tcpSocket->connectToHost(Options::pHD2Host(), Options::pHD2Port());
203
204 m_PHD2ReconnectCounter = 0;
205 stateTimer->start(PHD2_RECONNECT_TIMEOUT);
206 return true;
207
208 case EQUIPMENT_DISCONNECTED:
209 // Equipment disconnected from PHD2, request reconnection
210 connectEquipment(true);
211 return true;
212
213 case DISCONNECTING:
214 // Not supposed to interrupt a running disconnection
215 return false;
216
217 default:
218 return false;
219 }
220}
221
222void PHD2::ResetConnectionState()
223{
224 connection = DISCONNECTED;
225
226 // clear the outstanding and queued RPC requests
227 pendingRpcResultType = NO_RESULT;
228 rpcRequestQueue.clear();
229
230 starImageRequested = false;
231 isSettling = false;
232 isDitherActive = false;
233
234 ditherTimer->stop();
235 abortTimer->stop();
236
237 tcpSocket->disconnect(this);
238
239 emit newStatus(GUIDE_DISCONNECTED);
240}
241
242bool PHD2::Disconnect()
243{
244 switch (connection)
245 {
246 case EQUIPMENT_CONNECTED:
247 emit newLog(i18n("Aborting any capture before disconnecting equipment..."));
248 abort();
249 connection = DISCONNECTING;
250 break;
251
252 case CONNECTED:
253 case CONNECTING:
254 case EQUIPMENT_DISCONNECTED:
255 stateTimer->stop();
256 tcpSocket->disconnectFromHost();
257 ResetConnectionState();
258 if (tcpSocket->state() != QAbstractSocket::UnconnectedState)
259 tcpSocket->waitForDisconnected(5000);
260 emit newLog(i18n("Disconnected from PHD2 Host: %1, on port %2.", Options::pHD2Host(), Options::pHD2Port()));
261 break;
262
263 case DISCONNECTING:
264 case DISCONNECTED:
265 break;
266 }
267
268 return true;
269}
270
271void PHD2::displayError(QAbstractSocket::SocketError socketError)
272{
273 switch (socketError)
274 {
276 emit newLog(i18n("The host disconnected."));
277 break;
279 emit newLog(i18n("The host was not found. Please check the host name and port settings in Guide options."));
280 break;
282 emit newLog(i18n("The connection was refused by the peer. Make sure the PHD2 is running, and check that "
283 "the host name and port settings are correct."));
284 break;
285 default:
286 emit newLog(i18n("The following error occurred: %1.", tcpSocket->errorString()));
287 }
288
289 ResetConnectionState();
290
291 emit newStatus(GUIDE_DISCONNECTED);
292}
293
294void PHD2::readPHD2()
295{
296 while (!tcpSocket->atEnd() && tcpSocket->canReadLine())
297 {
298 QByteArray line = tcpSocket->readLine();
299 if (line.isEmpty())
300 continue;
301
302 QJsonParseError qjsonError;
303
304 QJsonDocument jdoc = QJsonDocument::fromJson(line, &qjsonError);
305
306 if (qjsonError.error != QJsonParseError::NoError)
307 {
308 emit newLog(i18n("PHD2: invalid response received: %1", QString(line)));
309 emit newLog(i18n("PHD2: JSON error: %1", qjsonError.errorString()));
310 continue;
311 }
312
313 QJsonObject jsonObj = jdoc.object();
314
315 if (jsonObj.contains("Event"))
316 processPHD2Event(jsonObj, line);
317 else if (jsonObj.contains("error"))
318 processPHD2Error(jsonObj, line);
319 else if (jsonObj.contains("result"))
320 processPHD2Result(jsonObj, line);
321 }
322}
323
324void PHD2::processPHD2Event(const QJsonObject &jsonEvent, const QByteArray &line)
325{
326 if (Options::verboseLogging())
327 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: event:" << line;
328
329 QString eventName = jsonEvent["Event"].toString();
330
331 if (!events.contains(eventName))
332 {
333 emit newLog(i18n("Unknown PHD2 event: %1", eventName));
334 return;
335 }
336
337 event = events.value(eventName);
338
339 switch (event)
340 {
341 case Version:
342 emit newLog(i18n("PHD2: Version %1", jsonEvent["PHDVersion"].toString()));
343 break;
344
345 case CalibrationComplete:
346 emit newLog(i18n("PHD2: Calibration Complete."));
347 emit newStatus(Ekos::GUIDE_CALIBRATION_SUCCESS);
348 break;
349
350 case StartGuiding:
351 updateGuideParameters();
352 requestCurrentEquipmentUpdate();
353 // Do not report guiding as started because it will start scheduled capture before guiding is settled
354 // just print the log message and GUIDE_STARTED status will be set in SettleDone
355 // phd2 will always send SettleDone event
356 emit newLog(i18n("PHD2: Waiting for guiding to settle."));
357 break;
358
359 case Paused:
360 handlePHD2AppState(PAUSED);
361 break;
362
363 case StartCalibration:
364 handlePHD2AppState(CALIBRATING);
365 break;
366
367 case AppState:
368 // AppState is the last of the initial messages received when we first connect to PHD2
369 processPHD2State(jsonEvent["State"].toString());
370 // if the equipment is not already connected, then try to connect it.
371 if (connection == CONNECTING)
372 {
373 emit newLog("PHD2: Connecting equipment and external guider...");
374 connectEquipment(true);
375 }
376 break;
377
378 case CalibrationFailed:
379 emit newLog(i18n("PHD2: Calibration Failed (%1).", jsonEvent["Reason"].toString()));
380 handlePHD2AppState(STOPPED);
381 break;
382
383 case CalibrationDataFlipped:
384 emit newLog(i18n("Calibration Data Flipped."));
385 break;
386
387 case LoopingExposures:
388 handlePHD2AppState(LOOPING);
389 break;
390
391 case LoopingExposuresStopped:
392 handlePHD2AppState(STOPPED);
393 break;
394
395 case Calibrating:
396 case Settling:
397 case SettleBegin:
398 //This can happen for guiding or for dithering. A Settle done event will arrive when it finishes.
399 break;
400
401 case SettleDone:
402 {
403 emit newLog(i18n("PHD2: SettleDone."));
404
405 // guiding stopped during dithering
406 if (state == PHD2::STOPPED)
407 return;
408
409 bool error = false;
410
411 if (jsonEvent["Status"].toInt() != 0)
412 {
413 error = true;
414 emit newLog(i18n("PHD2: Settling failed (%1).", jsonEvent["Error"].toString()));
415 }
416
417 bool wasDithering = isDitherActive;
418
419 isDitherActive = false;
420 isSettling = false;
421
422 if (wasDithering)
423 {
424 ditherTimer->stop();
425 if (error && Options::ditherFailAbortsAutoGuide())
426 {
427 abort();
428 emit newStatus(GUIDE_DITHERING_ERROR);
429 }
430 else
431 {
432 if (error)
433 emit newLog(i18n("PHD2: There was a dithering error, but continue guiding."));
434
435 emit newStatus(Ekos::GUIDE_DITHERING_SUCCESS);
436 }
437 }
438 else
439 {
440 if (error)
441 {
442 emit newLog(i18n("PHD2: Settling failed, aborted."));
443 emit newStatus(GUIDE_ABORTED);
444 }
445 else
446 {
447 // settle completed after "guide" command
448 emit newLog(i18n("PHD2: Settling complete, Guiding Started."));
449 emit newStatus(GUIDE_GUIDING);
450 }
451 }
452 }
453 break;
454
455 case StarSelected:
456 handlePHD2AppState(SELECTED);
457 break;
458
459 case StarLost:
460 // If we lost the guide star, let the state and abort timers update our state
461 handlePHD2AppState(LOSTLOCK);
462 break;
463
464 case GuidingStopped:
465 handlePHD2AppState(STOPPED);
466 break;
467
468 case Resumed:
469 handlePHD2AppState(GUIDING);
470 break;
471
472 case GuideStep:
473 {
474 // If we lost the guide star, let the state timer update our state
475 // Sometimes PHD2 is actually not guiding at that time, so we'll either resume or abort
476 if (state == LOSTLOCK)
477 emit newLog(i18n("PHD2: Star found, guiding is resuming..."));
478
479 if (isDitherActive)
480 return;
481
482 double diff_ra_pixels, diff_de_pixels, diff_ra_arcsecs, diff_de_arcsecs, pulse_ra, pulse_dec, snr;
483 QString RADirection, DECDirection;
484 diff_ra_pixels = jsonEvent["RADistanceRaw"].toDouble();
485 diff_de_pixels = jsonEvent["DECDistanceRaw"].toDouble();
486 pulse_ra = jsonEvent["RADuration"].toDouble();
487 pulse_dec = jsonEvent["DECDuration"].toDouble();
488 RADirection = jsonEvent["RADirection"].toString();
489 DECDirection = jsonEvent["DECDirection"].toString();
490 snr = jsonEvent["SNR"].toDouble();
491
492 if (RADirection == "East")
493 //If the pulse was to the east, it should have a negative sign.
494 //(Tracking pulse has to be decreased.)
495 pulse_ra = -pulse_ra;
496 if (DECDirection == "South")
497 //If the pulse was to the south, it should have a negative sign.
498 pulse_dec = -pulse_dec;
499
500 //If the pixelScale is properly set from PHD2, the second block of code is not needed,
501 //but if not, we will attempt to calculate the ra and dec error without it.
502 if (pixelScale != 0)
503 {
504 diff_ra_arcsecs = diff_ra_pixels * pixelScale;
505 diff_de_arcsecs = diff_de_pixels * pixelScale;
506 }
507 else
508 {
509 diff_ra_arcsecs = 206.26480624709 * diff_ra_pixels * ccdPixelSizeX / mountFocalLength;
510 diff_de_arcsecs = 206.26480624709 * diff_de_pixels * ccdPixelSizeY / mountFocalLength;
511 }
512
513 if (std::isfinite(snr))
514 emit newSNR(snr);
515
516 if (std::isfinite(diff_ra_arcsecs) && std::isfinite(diff_de_arcsecs))
517 {
518 // Always add to error log
519 errorLog.append(QPointF(diff_ra_arcsecs, diff_de_arcsecs));
520 if(errorLog.size() > 100)
521 errorLog.remove(0);
522
523 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: Error log size:" << errorLog.size();
524
525 // diff_xx_arcsecs is saved as STAR drift in the camera sensor coordinate system.
526 // To get these values in the RADEC system they have to be negated.
527 // But PHD2 displays the MOUNT drift and hence the values have to be negated once more!
528 // Guide graph in EKOS handles the reversed direction of RA-coordinate.
529 emit newAxisDelta(diff_ra_arcsecs, diff_de_arcsecs);
530 emit newAxisPulse(pulse_ra, pulse_dec);
531
532 // Does PHD2 real a sky background or num-stars measure?
533 // analyze.cpp uses only one RA/DEC-axis (up:+, down:-), hence RA is negated
534 // to handle the reversed direction of RA-coordinate.
535 emit guideStats(-diff_ra_arcsecs, diff_de_arcsecs, pulse_ra, pulse_dec,
536 std::isfinite(snr) ? snr : 0, 0, 0);
537
538 // Calculate population standard deviation (sigma) like PHD2 does
539 // PHD2 uses the formula: sqrt((n * sum(y²) - sum(y)²) / (n * n))
540
541 // Convert to pixels first if we have a valid pixel scale
542 QVector<QPointF> pixelErrorLog;
543 if (pixelScale > 0)
544 {
545 for (auto &point : errorLog)
546 {
547 // Convert from arcseconds to pixels
548 pixelErrorLog.append(QPointF(point.x() / pixelScale, point.y() / pixelScale));
549 }
550 }
551 else
552 {
553 // If no pixel scale, just use the original values
554 pixelErrorLog = errorLog;
555 }
556
557 // Calculate using the same formula as PHD2's GetPopulationSigma()
558 double n = pixelErrorLog.size();
559
560 // Calculate sums for RA
561 double sumY_RA = 0.0;
562 double sumYSq_RA = 0.0;
563 for (auto &point : pixelErrorLog)
564 {
565 sumY_RA += point.x();
566 sumYSq_RA += point.x() * point.x();
567 }
568
569 // Calculate sums for DEC
570 double sumY_DEC = 0.0;
571 double sumYSq_DEC = 0.0;
572 for (auto &point : pixelErrorLog)
573 {
574 sumY_DEC += point.y();
575 sumYSq_DEC += point.y() * point.y();
576 }
577
578 // Calculate population sigma using PHD2's formula
579 double ra_sigma_pixels = 0.0;
580 double de_sigma_pixels = 0.0;
581
582 if (n > 1)
583 {
584 double variance_RA = (n * sumYSq_RA - sumY_RA * sumY_RA) / (n * n);
585 double variance_DEC = (n * sumYSq_DEC - sumY_DEC * sumY_DEC) / (n * n);
586
587 if (variance_RA >= 0.0)
588 ra_sigma_pixels = sqrt(variance_RA);
589
590 if (variance_DEC >= 0.0)
591 de_sigma_pixels = sqrt(variance_DEC);
592 }
593
594 // Convert back to arcseconds for display
595 double ra_sigma_arcsec = (pixelScale > 0) ? ra_sigma_pixels * pixelScale : ra_sigma_pixels;
596 double de_sigma_arcsec = (pixelScale > 0) ? de_sigma_pixels * pixelScale : de_sigma_pixels;
597
598 // Emit the values in arcseconds
599 emit newAxisSigma(ra_sigma_arcsec, de_sigma_arcsec);
600 }
601 //Note that if it is receiving full size remote images, it should not get the guide star image.
602 //But if it is not getting the full size images, or if the current camera is not in Ekos, it should get the guide star image
603 //If we are getting the full size image, we will want to know the lock position for the image that loads in the viewer.
604 if ( Options::guideSubframe() || currentCameraIsNotInEkos )
605 requestStarImage(32); //This requests a star image for the guide view. 32 x 32 pixels
606 else
607 requestLockPosition();
608 }
609 break;
610
611 case GuidingDithered:
612 break;
613
614 case LockPositionSet:
615 handlePHD2AppState(SELECTED);
616 break;
617
618 case LockPositionLost:
619 handlePHD2AppState(LOSTLOCK);
620 break;
621
622 case Alert:
623 emit newLog(i18n("PHD2 %1: %2", jsonEvent["Type"].toString(), jsonEvent["Msg"].toString()));
624 break;
625
626 case GuideParamChange:
627 case ConfigurationChange:
628 //Don't do anything for now, might change this later.
629 //Some Possible Parameter Names:
630 //Backlash comp enabled, Backlash comp amount,
631 //For Each Axis: MinMove, Max Duration,
632 //PPEC aggressiveness, PPEC prediction weight,
633 //Resist switch minimum motion, Resist switch aggression,
634 //Low-pass minimum move, Low-pass slope weight,
635 //Low-pass2 minimum move, Low-pass2 aggressiveness,
636 //Hysteresis hysteresis, Hysteresis aggression
637 break;
638
639 }
640}
641
642void PHD2::processPHD2State(const QString &phd2State)
643{
644 if (phd2State == "Stopped")
645 handlePHD2AppState(STOPPED);
646 else if (phd2State == "Selected")
647 handlePHD2AppState(SELECTED);
648 else if (phd2State == "Calibrating")
649 handlePHD2AppState(CALIBRATING);
650 else if (phd2State == "Guiding")
651 handlePHD2AppState(GUIDING);
652 else if (phd2State == "LostLock")
653 handlePHD2AppState(LOSTLOCK);
654 else if (phd2State == "Paused")
655 handlePHD2AppState(PAUSED);
656 else if (phd2State == "Looping")
657 handlePHD2AppState(LOOPING);
658 else emit newLog(QString("PHD2: Unsupported app state ") + phd2State + ".");
659}
660
661void PHD2::handlePHD2AppState(PHD2State newstate)
662{
663 // do not handle the same state twice
664 if (state == newstate)
665 return;
666
667 switch (newstate)
668 {
669 case STOPPED:
670 switch (state)
671 {
672 case CALIBRATING:
673 //emit newLog(i18n("PHD2: Calibration Failed (%1).", jsonEvent["Reason"].toString()));
674 emit newStatus(Ekos::GUIDE_CALIBRATION_ERROR);
675 break;
676 case LOOPING:
677 emit newLog(i18n("PHD2: Looping Exposures Stopped."));
678 emit newStatus(Ekos::GUIDE_IDLE);
679 break;
680 case GUIDING:
681 case LOSTLOCK:
682 emit newLog(i18n("PHD2: Guiding Stopped."));
683 emit newStatus(Ekos::GUIDE_ABORTED);
684 break;
685 default:
686 if (connection == DISCONNECTING)
687 {
688 emit newLog("PHD2: Disconnecting equipment and external guider...");
689 connectEquipment(false);
690 }
691 break;
692 }
693 break;
694
695 case SELECTED:
696 switch (state)
697 {
698 case STOPPED:
699 case CALIBRATING:
700 case GUIDING:
701 emit newLog(i18n("PHD2: Lock Position Set."));
702 if (isSettling)
703 {
704 newstate = CALIBRATING;
705 emit newStatus(Ekos::GUIDE_CALIBRATING);
706 }
707 break;
708 case DITHERING:
709 // do nothing, this is the initial step in PHD2 for dithering
710 break;
711 default:
712 emit newLog(i18n("PHD2: Star Selected."));
713 emit newStatus(GUIDE_STAR_SELECT);
714 }
715 break;
716
717 case GUIDING:
718 switch (state)
719 {
720 case DITHERING:
721 emit newLog(i18n("PHD2: Guiding...waiting for settle..."));
722 break;
723
724 case PAUSED:
725 emit newLog(i18n("PHD2: Dithering successful."));
726 abortTimer->stop();
727 ditherTimer->stop(); // stop immediately, since pausing could interrupt settling
728 emit newStatus(Ekos::GUIDE_DITHERING_SUCCESS);
729 break;
730 default:
731 emit newLog(i18n("PHD2: Guiding started."));
732 abortTimer->stop();
733 emit newStatus(Ekos::GUIDE_GUIDING);
734 break;
735 }
736 break;
737
738 case LOSTLOCK:
739 switch (state)
740 {
741 case CALIBRATING:
742 emit newLog(i18n("PHD2: Lock Position Lost, continuing calibration."));
743 // Don't be paranoid, accept star-lost events during calibration and trust PHD2 to complete
744 //newstate = STOPPED;
745 //emit newStatus(Ekos::GUIDE_CALIBRATION_ERROR);
746 break;
747 case GUIDING:
748 emit newLog(i18n("PHD2: Star Lost. Trying to reacquire for %1s.", Options::guideLostStarTimeout()));
749 abortTimer->start(static_cast<int>(Options::guideLostStarTimeout()) * 1000);
750 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: Lost star timeout started (" << Options::guideLostStarTimeout() << " sec)";
751 emit newStatus(Ekos::GUIDE_REACQUIRE);
752 break;
753 default:
754 emit newLog(i18n("PHD2: Lock Position Lost."));
755 break;
756 }
757 break;
758
759 case PAUSED:
760 emit newLog(i18n("PHD2: Guiding paused."));
761 emit newStatus(GUIDE_SUSPENDED);
762 break;
763
764 case CALIBRATING:
765 emit newLog(i18n("PHD2: Calibrating, timing out in %1s.", Options::guideCalibrationTimeout()));
766 abortTimer->start(static_cast<int>(Options::guideCalibrationTimeout()) * 1000);
767 emit newStatus(GUIDE_CALIBRATING);
768 break;
769
770 case LOOPING:
771 switch (state)
772 {
773 case CALIBRATING:
774 emit newLog(i18n("PHD2: Calibration turned to looping, failed."));
775 emit newStatus(GUIDE_CALIBRATION_ERROR);
776 break;
777 default:
778 emit newLog(i18n("PHD2: Looping Exposures."));
779 emit newStatus(GUIDE_LOOPING);
780 break;
781 }
782 break;
783 case DITHERING:
784 emit newLog(i18n("PHD2: Dithering started."));
785 emit newStatus(GUIDE_DITHERING);
786 break;
787 }
788
789 state = newstate;
790}
791
792void PHD2::processPHD2Result(const QJsonObject &jsonObj, const QByteArray &line)
793{
794 PHD2ResultType resultType = takeRequestFromList(jsonObj);
795
796 if (resultType == STAR_IMAGE)
797 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: received star image response, id" <<
798 jsonObj["id"].toInt(); // don't spam the log with image data
799 else
800 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: response:" << line;
801
802 switch (resultType)
803 {
804 case NO_RESULT:
805 //Ekos didn't ask for this result?
806 break;
807
808 case CAPTURE_SINGLE_FRAME: //capture_single_frame
809 break;
810
811 case CLEAR_CALIBRATION_COMMAND_RECEIVED: //clear_calibration
812 emit newLog(i18n("PHD2: Calibration is cleared"));
813 break;
814
815 case DITHER_COMMAND_RECEIVED: //dither
816 handlePHD2AppState(DITHERING);
817 break;
818
819 //find_star
820 //flip_calibration
821 //get_algo_param_names
822 //get_algo_param
823
824 case APP_STATE_RECEIVED: //get_app_state
825 {
826 QString state = jsonObj["State"].toString();
827 if (state.isEmpty())
828 state = jsonObj["result"].toString();
829 if (state.isEmpty())
830 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: received unsupported app state";
831 else
832 processPHD2State(state);
833 }
834 break;
835
836 //get_calibrated
837 //get_calibration_data
838
839 case IS_EQUIPMENT_CONNECTED: //get_connected
840 {
841 bool isConnected = jsonObj["result"].toBool();
842 switch (connection)
843 {
844 case CONNECTING:
845 // We just plugged in server, request equipment connection if needed
846 if (isConnected)
847 {
848 connection = CONNECTED;
849 setEquipmentConnected();
850 }
851 else connectEquipment(true);
852 break;
853
854 case CONNECTED:
855 // We were waiting for equipment to be connected after plugging in server
856 if (isConnected)
857 setEquipmentConnected();
858 break;
859
860 case DISCONNECTING:
861 // We were waiting for equipment to be disconnected before unplugging from server
862 if (!isConnected)
863 {
864 connection = EQUIPMENT_DISCONNECTED;
865 Disconnect();
866 }
867 else connectEquipment(false);
868 break;
869
870 case EQUIPMENT_CONNECTED:
871 // Equipment was disconnected from PHD2 side, so notify clients and wait.
872 if (!isConnected)
873 {
874 // TODO: setEquipmentDisconnected()
875 connection = EQUIPMENT_DISCONNECTED;
876 emit newStatus(Ekos::GUIDE_DISCONNECTED);
877 }
878 break;
879
880 case DISCONNECTED:
881 case EQUIPMENT_DISCONNECTED:
882 // Equipment was connected from PHD2 side, so notify clients and wait.
883 if (isConnected)
884 setEquipmentConnected();
885 break;
886 }
887 }
888 break;
889
890 //get_cooler_status
891 case GET_CURRENT_EQUIPMENT: //get_current_equipment
892 {
893 QJsonObject equipObject = jsonObj["result"].toObject();
894 currentCamera = equipObject["camera"].toObject()["name"].toString();
895 currentMount = equipObject["mount"].toObject()["name"].toString();
896 currentAuxMount = equipObject["aux_mount"].toObject()["name"].toString();
897
898 emit guideEquipmentUpdated();
899
900 break;
901 }
902
903
904 case DEC_GUIDE_MODE: //get_dec_guide_mode
905 {
906 QString mode = jsonObj["result"].toString();
907 Ekos::Manager::Instance()->guideModule()->updateDirectionsFromPHD2(mode);
908 emit newLog(i18n("PHD2: DEC Guide Mode is Set to: %1", mode));
909 }
910 break;
911
912
913 case EXPOSURE_TIME: //get_exposure
914 {
915 int exposurems = jsonObj["result"].toInt();
916 double exposureTime = exposurems / 1000.0;
917 Ekos::Manager::Instance()->guideModule()->setExposure(exposureTime);
918 emit newLog(i18n("PHD2: Exposure Time set to: ") + QString::number(exposureTime, 'f', 2));
919 break;
920 }
921
922
923 case EXPOSURE_DURATIONS: //get_exposure_durations
924 {
925 QVariantList exposureListArray = jsonObj["result"].toArray().toVariantList();
926 logValidExposureTimes = i18n("PHD2: Valid Exposure Times: Auto, ");
927 QList<double> values;
928 for(int i = 1; i < exposureListArray.size();
929 i ++) //For some reason PHD2 has a negative exposure time of 1 at the start of the array?
930 values << exposureListArray.at(i).toDouble() / 1000.0; //PHD2 reports in ms.
931 logValidExposureTimes += Ekos::Manager::Instance()->guideModule()->setRecommendedExposureValues(values);
932 emit newLog(logValidExposureTimes);
933 break;
934 }
935 case LOCK_POSITION: //get_lock_position
936 {
937 if(jsonObj["result"].toArray().count() == 2)
938 {
939 double x = jsonObj["result"].toArray().at(0).toDouble();
940 double y = jsonObj["result"].toArray().at(1).toDouble();
941 QVector3D newStarCenter(x, y, 0);
942 emit newStarPosition(newStarCenter, true);
943
944 //This is needed so that PHD2 sends the new star pixmap when
945 //remote images are enabled.
946 emit newStarPixmap(m_GuideFrame->getTrackingBoxPixmap());
947 }
948 break;
949 }
950 //get_lock_shift_enabled
951 //get_lock_shift_params
952 //get_paused
953
954 case PIXEL_SCALE: //get_pixel_scale
955 pixelScale = jsonObj["result"].toDouble();
956 if (pixelScale == 0)
957 emit newLog(i18n("PHD2: Please set CCD and telescope parameters in PHD2, Pixel Scale is invalid."));
958 else
959 emit newLog(i18n("PHD2: Pixel Scale is %1 arcsec per pixel", QString::number(pixelScale, 'f', 2)));
960 break;
961
962 //get_profile
963 //get_profiles
964 //get_search_region
965 //get_sensor_temperature
966
967 case STAR_IMAGE: //get_star_image
968 {
969 starImageRequested = false;
970 QJsonObject jsonResult = jsonObj["result"].toObject();
971 processStarImage(jsonResult);
972 break;
973 }
974
975 //get_use_subframes
976
977 case GUIDE_COMMAND_RECEIVED: //guide
978 if (0 != jsonObj["result"].toInt(0))
979 {
980 emit newLog("PHD2: Guide command was rejected.");
981 handlePHD2AppState(STOPPED);
982 }
983 break;
984
985 //guide_pulse
986
987 case LOOP: //loop
988 handlePHD2AppState(jsonObj["result"].toBool() ? LOOPING : STOPPED);
989 break;
990
991 //save_image
992 //set_algo_param
993
994 case CONNECTION_RESULT: //set_connected
995 checkIfEquipmentConnected();
996 break;
997
998 case SET_DEC_GUIDE_MODE_COMMAND_RECEIVED: //set_dec_guide_mode
999 checkDEGuideMode();
1000 break;
1001
1002 case SET_EXPOSURE_COMMAND_RECEIVED: //set_exposure
1003 requestExposureTime(); //This will check what it was set to and print a message as to what it is.
1004 break;
1005
1006 case SET_LOCK_POSITION: //set_lock_position
1007 handlePHD2AppState(SELECTED);
1008 break;
1009
1010 //set_lock_shift_enabled
1011 //set_lock_shift_params
1012
1013 case SET_PAUSED_COMMAND_RECEIVED: //set_paused
1014 handlePHD2AppState(PAUSED);
1015 break;
1016 //set_profile
1017 //shutdown
1018
1019 case STOP_CAPTURE_COMMAND_RECEIVED: //stop_capture
1020 handlePHD2AppState(STOPPED);
1021 //emit newStatus(GUIDE_ABORTED);
1022 break;
1023 }
1024
1025 // send the next pending call
1026 sendNextRpcCall();
1027}
1028
1029void PHD2::processPHD2Error(const QJsonObject &jsonError, const QByteArray &line)
1030{
1031 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: error:" << line;
1032
1033 QJsonObject jsonErrorObject = jsonError["error"].toObject();
1034
1035 PHD2ResultType resultType = takeRequestFromList(jsonError);
1036
1037 // This means the user mistakenly entered an invalid exposure time.
1038 switch (resultType)
1039 {
1040 case SET_EXPOSURE_COMMAND_RECEIVED:
1041 emit newLog(logValidExposureTimes); //This will let the user know the valid exposure durations
1042 QTimer::singleShot(300, [ = ] {requestExposureTime();}); //This will reset the Exposure time in Ekos to PHD2's current exposure time after a third of a second.
1043 break;
1044
1045 case CONNECTION_RESULT:
1046 connection = EQUIPMENT_DISCONNECTED;
1047 emit newStatus(Ekos::GUIDE_DISCONNECTED);
1048 break;
1049
1050 case DITHER_COMMAND_RECEIVED:
1051 ditherTimer->stop();
1052 isSettling = false;
1053 isDitherActive = false;
1054 emit newStatus(GUIDE_DITHERING_ERROR);
1055
1056 if (Options::ditherFailAbortsAutoGuide())
1057 {
1058 abort();
1059 emit newLog("PHD2: failing after dithering aborts.");
1060 emit newStatus(GUIDE_ABORTED);
1061 }
1062 else
1063 {
1064 // !FIXME-ag why is this trying to resume (un-pause)?
1065 resume();
1066 }
1067 break;
1068
1069 case GUIDE_COMMAND_RECEIVED:
1070 isSettling = false;
1071 break;
1072
1073 default:
1074 emit newLog(i18n("PHD2 Error: unhandled '%1'", jsonErrorObject["message"].toString()));
1075 break;
1076 }
1077
1078 // send the next pending call
1079 sendNextRpcCall();
1080}
1081
1082//These methods process the Star Images the PHD2 provides
1083
1084void PHD2::setGuideView(const QSharedPointer<FITSView> &guideView)
1085{
1086 m_GuideFrame = guideView;
1087}
1088
1089void PHD2::processStarImage(const QJsonObject &jsonStarFrame)
1090{
1091 //The width and height of the received PHD2 Star Image
1092 int width = jsonStarFrame["width"].toInt();
1093 int height = jsonStarFrame["height"].toInt();
1094
1095 //This section sets up the FITS File
1096 fitsfile *fptr = nullptr;
1097 int status = 0;
1098 long fpixel = 1, naxis = 2, nelements, exposure;
1099 long naxes[2] = { width, height };
1100 char error_status[512] = {0};
1101
1102 void* fits_buffer = nullptr;
1103 size_t fits_buffer_size = 0;
1104 if (fits_create_memfile(&fptr, &fits_buffer, &fits_buffer_size, 4096, realloc, &status))
1105 {
1106 qCWarning(KSTARS_EKOS_GUIDE) << "fits_create_file failed:" << error_status;
1107 return;
1108 }
1109
1110 if (fits_create_img(fptr, USHORT_IMG, naxis, naxes, &status))
1111 {
1112 qCWarning(KSTARS_EKOS_GUIDE) << "fits_create_img failed:" << error_status;
1113 status = 0;
1114 fits_close_file(fptr, &status);
1115 free(fits_buffer);
1116 return;
1117 }
1118
1119 //Note, this is made up. If you want the actual exposure time, you have to request it from PHD2
1120 exposure = 1;
1121 fits_update_key(fptr, TLONG, "EXPOSURE", &exposure, "Total Exposure Time", &status);
1122
1123 //This section takes the Pixels from the JSON Document
1124 //Then it converts from base64 to a QByteArray
1125 //Then it creates a datastream from the QByteArray to the pixel array for the FITS File
1126 QByteArray converted = QByteArray::fromBase64(jsonStarFrame["pixels"].toString().toLocal8Bit());
1127
1128 //This finishes up and closes the FITS file
1129 nelements = naxes[0] * naxes[1];
1130 if (fits_write_img(fptr, TUSHORT, fpixel, nelements, converted.data(), &status))
1131 {
1132 fits_get_errstatus(status, error_status);
1133 qCWarning(KSTARS_EKOS_GUIDE) << "fits_write_img failed:" << error_status;
1134 status = 0;
1135 fits_close_file(fptr, &status);
1136 free(fits_buffer);
1137 return;
1138 }
1139
1140 if (fits_flush_file(fptr, &status))
1141 {
1142 fits_get_errstatus(status, error_status);
1143 qCWarning(KSTARS_EKOS_GUIDE) << "fits_flush_file failed:" << error_status;
1144 status = 0;
1145 fits_close_file(fptr, &status);
1146 free(fits_buffer);
1147 return;
1148 }
1149
1150 if (fits_close_file(fptr, &status))
1151 {
1152 fits_get_errstatus(status, error_status);
1153 qCWarning(KSTARS_EKOS_GUIDE) << "fits_close_file failed:" << error_status;
1154 free(fits_buffer);
1155 return;
1156 }
1157
1158 //This loads the FITS file in the Guide FITSView
1159 //Then it updates the Summary Screen
1160 QSharedPointer<FITSData> fdata;
1161 QByteArray buffer = QByteArray::fromRawData(reinterpret_cast<char *>(fits_buffer), fits_buffer_size);
1162 fdata.reset(new FITSData(), &QObject::deleteLater);
1163 fdata->setExtension(QString("fits"));
1164 fdata->loadFromBuffer(buffer);
1165 free(fits_buffer);
1166 m_GuideFrame->loadData(fdata);
1167
1168 m_GuideFrame->updateFrame();
1169 m_GuideFrame->setTrackingBox(QRect(0, 0, width, height));
1170 emit newStarPixmap(m_GuideFrame->getTrackingBoxPixmap());
1171}
1172
1173void PHD2::setEquipmentConnected()
1174{
1175 if (connection != EQUIPMENT_CONNECTED)
1176 {
1177 setConnectedRetries = 0;
1178 connection = EQUIPMENT_CONNECTED;
1179 emit newStatus(Ekos::GUIDE_CONNECTED);
1180 updateGuideParameters();
1181 requestExposureDurations();
1182 requestCurrentEquipmentUpdate();
1183 }
1184}
1185
1186void PHD2::updateGuideParameters()
1187{
1188 if (pixelScale == 0)
1189 requestPixelScale();
1190 requestExposureTime();
1191 checkDEGuideMode();
1192}
1193
1194//This section handles the methods/requests sent to PHD2, some are not implemented.
1195
1196//capture_single_frame
1197void PHD2::captureSingleFrame()
1198{
1199 sendPHD2Request("capture_single_frame");
1200}
1201
1202//clear_calibration
1203bool PHD2::clearCalibration()
1204{
1205 if (connection != EQUIPMENT_CONNECTED)
1206 {
1207 emit newLog(i18n("PHD2 Error: Equipment not connected."));
1208 emit newStatus(Ekos::GUIDE_ABORTED);
1209 return false;
1210 }
1211
1212 QJsonArray args;
1213 //This instructs PHD2 which calibration to clear.
1214 args << "mount";
1215 sendPHD2Request("clear_calibration", args);
1216
1217 return true;
1218}
1219
1220//dither
1221bool PHD2::dither(double pixels)
1222{
1223 if (connection != EQUIPMENT_CONNECTED)
1224 {
1225 emit newLog(i18n("PHD2 Error: Equipment not connected."));
1226 emit newStatus(Ekos::GUIDE_ABORTED);
1227 return false;
1228 }
1229
1230 if (isSettling)
1231 {
1232 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: ignoring dither requested while already settling";
1233
1234 if (!isDitherActive)
1235 {
1236 // act like we just dithered so we get the appropriate
1237 // effects after the settling completes
1238 handlePHD2AppState(DITHERING);
1239 isDitherActive = true;
1240 }
1241 return true;
1242 }
1243
1244 QJsonArray args;
1245 QJsonObject settle;
1246
1247 int ditherTimeout = static_cast<int>(Options::ditherTimeout());
1248
1249 settle.insert("pixels", static_cast<double>(Options::ditherThreshold()));
1250 settle.insert("time", static_cast<int>(Options::ditherSettle()));
1251 settle.insert("timeout", ditherTimeout);
1252
1253 // Pixels
1254 args << pixels;
1255 // RA Only?
1256 args << false;
1257 // Settle
1258 args << settle;
1259
1260 isSettling = true;
1261 isDitherActive = true;
1262
1263 // PHD2 will send a SettleDone event shortly after the settling
1264 // timeout in PHD2. We don't really need a timer here, but we'll
1265 // set one anyway (belt and suspenders). Make sure to give an
1266 // extra time allowance since PHD2 won't report its timeout until
1267 // the completion of the next guide exposure after the timeout
1268 // period expires.
1269 enum { TIMEOUT_EXTRA_SECONDS = 60 }; // at least as long as any reasonable guide exposure
1270 int millis = (ditherTimeout + TIMEOUT_EXTRA_SECONDS) * 1000;
1271 ditherTimer->start(millis);
1272
1273 sendPHD2Request("dither", args);
1274
1275 handlePHD2AppState(DITHERING);
1276
1277 return true;
1278}
1279
1280//find_star
1281//flip_calibration
1282//get_algo_param_names
1283//get_algo_param
1284
1285//get_app_state
1286void PHD2::requestAppState()
1287{
1288 sendPHD2Request("get_app_state");
1289}
1290
1291//get_calibrated
1292//get_calibration_data
1293
1294//get_connected
1295void PHD2::checkIfEquipmentConnected()
1296{
1297 sendPHD2Request("get_connected");
1298}
1299
1300//get_cooler_status
1301//get_current_equipment
1302void PHD2::requestCurrentEquipmentUpdate()
1303{
1304 sendPHD2Request("get_current_equipment");
1305}
1306
1307//get_dec_guide_mode
1308void PHD2::checkDEGuideMode()
1309{
1310 sendPHD2Request("get_dec_guide_mode");
1311}
1312
1313//get_exposure
1314void PHD2::requestExposureTime()
1315{
1316 sendPHD2Request("get_exposure");
1317}
1318
1319//get_exposure_durations
1320void PHD2::requestExposureDurations()
1321{
1322 sendPHD2Request("get_exposure_durations");
1323}
1324
1325//get_lock_position
1326void PHD2::requestLockPosition()
1327{
1328 sendPHD2Request("get_lock_position");
1329}
1330//get_lock_shift_enabled
1331//get_lock_shift_params
1332//get_paused
1333
1334//get_pixel_scale
1335void PHD2::requestPixelScale()
1336{
1337 sendPHD2Request("get_pixel_scale");
1338}
1339
1340//get_profile
1341//get_profiles
1342//get_search_region
1343//get_sensor_temperature
1344
1345//get_star_image
1346void PHD2::requestStarImage(int size)
1347{
1348 if (starImageRequested)
1349 {
1350 if (Options::verboseLogging())
1351 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: skip extra star image request";
1352 return;
1353 }
1354
1355 QJsonArray args2;
1356 args2 << size; // This is both the width and height.
1357 sendPHD2Request("get_star_image", args2);
1358
1359 starImageRequested = true;
1360}
1361
1362//get_use_subframes
1363
1364//guide
1365bool PHD2::guide()
1366{
1367 if (state == GUIDING)
1368 {
1369 emit newLog(i18n("PHD2: Guiding is already running."));
1370 emit newStatus(Ekos::GUIDE_GUIDING);
1371 return true;
1372 }
1373
1374 if (connection != EQUIPMENT_CONNECTED)
1375 {
1376 emit newLog(i18n("PHD2 Error: Equipment not connected."));
1377 emit newStatus(Ekos::GUIDE_ABORTED);
1378 return false;
1379 }
1380
1381 QJsonArray args;
1382 QJsonObject settle;
1383
1384 settle.insert("pixels", static_cast<double>(Options::ditherThreshold()));
1385 settle.insert("time", static_cast<int>(Options::ditherSettle()));
1386 settle.insert("timeout", static_cast<int>(Options::ditherTimeout()));
1387
1388 // Settle param
1389 args << settle;
1390 // Recalibrate param
1391 args << false;
1392
1393 errorLog.clear();
1394
1395 isSettling = true;
1396 sendPHD2Request("guide", args);
1397
1398 return true;
1399}
1400
1401//guide_pulse
1402//loop
1403void PHD2::loop()
1404{
1405 sendPHD2Request("loop");
1406}
1407//save_image
1408//set_algo_param
1409
1410//set_connected
1411void PHD2::connectEquipment(bool enable)
1412{
1413 if (connection == EQUIPMENT_CONNECTED && enable == true)
1414 return;
1415
1416 if (connection == EQUIPMENT_DISCONNECTED && enable == false)
1417 return;
1418
1419 if (setConnectedRetries++ > MAX_SET_CONNECTED_RETRIES)
1420 {
1421 setConnectedRetries = 0;
1422 connection = EQUIPMENT_DISCONNECTED;
1423 emit newStatus(Ekos::GUIDE_DISCONNECTED);
1424 return;
1425 }
1426
1427 pixelScale = 0 ;
1428
1429 QJsonArray args;
1430
1431 // connected = enable
1432 args << enable;
1433
1434 if (enable)
1435 emit newLog(i18n("PHD2: Connecting Equipment. . ."));
1436 else
1437 emit newLog(i18n("PHD2: Disconnecting Equipment. . ."));
1438
1439 sendPHD2Request("set_connected", args);
1440}
1441
1442//set_dec_guide_mode
1443void PHD2::requestSetDEGuideMode(bool deEnabled, bool nEnabled,
1444 bool sEnabled) //Possible Settings Off, Auto, North, and South
1445{
1446 QJsonArray args;
1447
1448 if(deEnabled)
1449 {
1450 if(nEnabled && sEnabled)
1451 args << "Auto";
1452 else if(nEnabled)
1453 args << "North";
1454 else if(sEnabled)
1455 args << "South";
1456 else
1457 args << "Off";
1458 }
1459 else
1460 {
1461 args << "Off";
1462 }
1463
1464 sendPHD2Request("set_dec_guide_mode", args);
1465}
1466
1467//set_exposure
1468void PHD2::requestSetExposureTime(int time) //Note: time is in milliseconds
1469{
1470 QJsonArray args;
1471 args << time;
1472 sendPHD2Request("set_exposure", args);
1473}
1474
1475//set_lock_position
1476void PHD2::setLockPosition(double x, double y)
1477{
1478 // Note: false will mean if a guide star is near the coordinates, it will use that.
1479 QJsonArray args;
1480 args << x << y << false;
1481 sendPHD2Request("set_lock_position", args);
1482}
1483//set_lock_shift_enabled
1484//set_lock_shift_params
1485
1486//set_paused
1487bool PHD2::suspend()
1488{
1489 if (connection != EQUIPMENT_CONNECTED)
1490 {
1491 emit newLog(i18n("PHD2 Error: Equipment not connected."));
1492 emit newStatus(Ekos::GUIDE_ABORTED);
1493 return false;
1494 }
1495
1496 QJsonArray args;
1497
1498 // Paused param
1499 args << true;
1500 // FULL param
1501 args << "full";
1502
1503 sendPHD2Request("set_paused", args);
1504
1505 if (abortTimer->isActive())
1506 {
1507 // Avoid that the a preceding lost star event leads to an abort while guiding is suspended.
1508 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: Lost star timeout cancelled.";
1509 abortTimer->stop();
1510 }
1511
1512 return true;
1513}
1514
1515//set_paused (also)
1516bool PHD2::resume()
1517{
1518 if (connection != EQUIPMENT_CONNECTED)
1519 {
1520 emit newLog(i18n("PHD2 Error: Equipment not connected."));
1521 emit newStatus(Ekos::GUIDE_ABORTED);
1522 return false;
1523 }
1524
1525 QJsonArray args;
1526
1527 // Paused param
1528 args << false;
1529
1530 sendPHD2Request("set_paused", args);
1531
1532 if (state == LOSTLOCK)
1533 {
1534 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: Lost star timeout restarted.";
1535 abortTimer->start(static_cast<int>(Options::guideLostStarTimeout()) * 1000);
1536 }
1537
1538 return true;
1539}
1540
1541//set_profile
1542//shutdown
1543
1544//stop_capture
1545bool PHD2::abort()
1546{
1547 if (connection != EQUIPMENT_CONNECTED)
1548 {
1549 emit newLog(i18n("PHD2 Error: Equipment not connected."));
1550 emit newStatus(Ekos::GUIDE_ABORTED);
1551 return false;
1552 }
1553
1554 abortTimer->stop();
1555
1556 sendPHD2Request("stop_capture");
1557 return true;
1558}
1559
1560//This method is not handled by PHD2
1561bool PHD2::calibrate()
1562{
1563 // We don't explicitly do calibration since it is done in the guide step by PHD2 anyway
1564 //emit newStatus(Ekos::GUIDE_CALIBRATION_SUCCESS);
1565 return true;
1566}
1567
1568//This is how information requests and commands for PHD2 are handled
1569
1570void PHD2::sendRpcCall(QJsonObject &call, PHD2ResultType resultType)
1571{
1572 assert(resultType != NO_RESULT); // should be a real request
1573 assert(pendingRpcResultType == NO_RESULT); // only one pending RPC call at a time
1574
1575 if (tcpSocket->state() == QTcpSocket::ConnectedState)
1576 {
1577 int rpcId = nextRpcId++;
1578 call.insert("id", rpcId);
1579
1580 QByteArray request = QJsonDocument(call).toJson(QJsonDocument::Compact);
1581
1582 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: request:" << request;
1583
1584 request.append("\r\n");
1585
1586 qint64 const n = tcpSocket->write(request);
1587
1588 if ((int) n == request.size())
1589 {
1590 // RPC call succeeded, remember ID and expected result type
1591 pendingRpcId = rpcId;
1592 pendingRpcResultType = resultType;
1593 }
1594 else
1595 {
1596 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: unexpected short write:" << n << "bytes of" << request.size();
1597 }
1598 }
1599}
1600
1601void PHD2::sendNextRpcCall()
1602{
1603 if (pendingRpcResultType != NO_RESULT)
1604 return; // a request is currently outstanding
1605
1606 if (rpcRequestQueue.empty())
1607 return; // no queued requests
1608
1609 RpcCall &call = rpcRequestQueue.front();
1610 sendRpcCall(call.call, call.resultType);
1611 rpcRequestQueue.pop_front();
1612}
1613
1614void PHD2::sendPHD2Request(const QString &method, const QJsonArray &args)
1615{
1616 assert(methodResults.contains(method));
1617
1618 PHD2ResultType resultType = methodResults[method];
1619
1620 QJsonObject jsonRPC;
1621
1622 jsonRPC.insert("jsonrpc", "2.0");
1623 jsonRPC.insert("method", method);
1624
1625 if (!args.empty())
1626 jsonRPC.insert("params", args);
1627
1628 if (pendingRpcResultType == NO_RESULT)
1629 {
1630 // no outstanding rpc call, send it right off
1631 sendRpcCall(jsonRPC, resultType);
1632 }
1633 else
1634 {
1635 // there is already an outstanding call, enqueue this call
1636 // until the prior call completes
1637
1638 if (Options::verboseLogging())
1639 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: defer call" << method;
1640
1641 rpcRequestQueue.push_back(RpcCall(jsonRPC, resultType));
1642 }
1643}
1644
1645PHD2::PHD2ResultType PHD2::takeRequestFromList(const QJsonObject &response)
1646{
1647 if (Q_UNLIKELY(!response.contains("id")))
1648 {
1649 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: ignoring unexpected response with no id";
1650 return NO_RESULT;
1651 }
1652
1653 int id = response["id"].toInt();
1654
1655 if (Q_UNLIKELY(id != pendingRpcId))
1656 {
1657 // RPC id mismatch -- this should never happen, something is
1658 // seriously wrong
1659 qCDebug(KSTARS_EKOS_GUIDE) << "PHD2: ignoring unexpected response with id" << id;
1660 return NO_RESULT;
1661 }
1662
1663 PHD2ResultType val = pendingRpcResultType;
1664 pendingRpcResultType = NO_RESULT;
1665 return val;
1666}
1667
1668}
Q_SCRIPTABLE Q_NOREPLY void setExposure(double value)
DBUS interface function.
Definition guide.cpp:1559
QString i18n(const char *text, const TYPE &arg...)
char * toString(const EngineQuery &query)
Ekos is an advanced Astrophotography tool for Linux.
Definition align.cpp:83
void error(QWidget *parent, const QString &text, const QString &title, const KGuiItem &buttonOk, Options options=Notify)
NETWORKMANAGERQT_EXPORT NetworkManager::Status status()
void errorOccurred(QAbstractSocket::SocketError socketError)
QByteArray & append(QByteArrayView data)
char * data()
QByteArray fromBase64(const QByteArray &base64, Base64Options options)
QByteArray fromRawData(const char *data, qsizetype size)
bool isEmpty() const const
qsizetype size() const const
void readyRead()
bool empty() const const
QJsonDocument fromJson(const QByteArray &json, QJsonParseError *error)
QJsonObject object() const const
bool contains(QLatin1StringView key) const const
iterator insert(QLatin1StringView key, const QJsonValue &value)
QJsonValue value(QLatin1StringView key) const const
QString errorString() const const
void append(QList< T > &&value)
qsizetype size() const const
QMetaObject::Connection connect(const QObject *sender, PointerToMemberFunction signal, Functor functor)
void deleteLater()
QString number(double n, char format, int precision)
void push_back(QChar ch)
UniqueConnection
void timeout()
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Mar 28 2025 11:57:24 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.