API
 
Loading...
Searching...
No Matches
mcp3008Ctrl.hpp
Go to the documentation of this file.
1/** \file mcp3008Ctrl.hpp
2 * \brief The MagAO-X MCP3008 Controller header file
3 *
4 * \ingroup mcp3008Ctrl_files
5 */
6
7#ifndef mcp3008Ctrl_hpp
8#define mcp3008Ctrl_hpp
9
10#include "../../libMagAOX/libMagAOX.hpp" //Note this is included on command line to trigger pch
11#include "../../magaox_git_version.h"
12#include "dependencies/MCP3008.h" // Included for adc.connect()
13
14/** \defgroup mcp3008Ctrl
15 * \brief The MagAO-X application to readout a MCP3008 A/D on a raspberry Pi.
16 *
17 * <a href="../handbook/operating/software/apps/XXXXXX.html">Application Documentation</a>
18 *
19 * \ingroup apps
20 *
21 */
22
23/** \defgroup mcp3008Ctrl_files
24 * \ingroup mcp3008Ctrl
25 */
26
27namespace MagAOX
28{
29namespace app
30{
31
32/// The MagAO-X MCP3008 Controller
33/**
34 * \ingroup mcp3008Ctrl
35 */
36class mcp3008Ctrl : public MagAOXApp<true>, public dev::frameGrabber<mcp3008Ctrl>, public dev::telemeter<mcp3008Ctrl>
37{
38
39 // Give the test harness access.
40 friend class mcp3008Ctrl_test;
41 friend class dev::frameGrabber<mcp3008Ctrl>;
42 friend class dev::telemeter<mcp3008Ctrl>;
43
46
48
49 static constexpr bool c_frameGrabber_flippable = false; /**< app:dev config to tell framegrabber these images
50 can not be flipped*/
51
52 protected:
53 /** \name Configurable Parameters
54 *@{
55 */
56
57 /** \todo @PARKER make this configurable DONE */
58 int m_numChannels{ 8 }; ///< The number of channels being read out.
59
60 ///@}
61
62 /** \todo @PARKER make this settable by INDI DONE*/
63 // Creating INDI property for desired fps
64 pcf::IndiProperty m_indiP_fps;
66 float m_fps{ 2000 }; ///< The target FPS
67
68 /** \todo @PARKER make this changed as needed */
69 float m_trigger{ 1e6 / m_fps }; ///< The trigger time to readout. Adjusts to match desired FPS.
70 float m_gain { .1 }; // Gain used to adjust trigger to keep at correct fps
71
73
74 std::chrono::time_point<std::chrono::high_resolution_clock> m_time_start;
75
76 std::vector<float> m_values; ///< The values read out from the chip
77
78 public:
79 /// Default c'tor.
81
82 /// D'tor, declared and defined for noexcept.
86
87 virtual void setupConfig();
88
89 /// Implementation of loadConfig logic, separated for testing.
90 /** This is called by loadConfig().
91 */
93 mx::app::appConfigurator &_config /**< [in] an application configuration
94 from which to load values*/ );
95
96 virtual void loadConfig();
97
98 /// Startup function
99 /**
100 *
101 */
102 virtual int appStartup();
103
104 /// Implementation of the FSM for mcp3008Ctrl.
105 /**
106 * \returns 0 on no critical error
107 * \returns -1 on an error requiring shutdown
108 */
109 virtual int appLogic();
110
111 /// Shutdown the app.
112 /**
113 *
114 */
115 virtual int appShutdown();
116
117 /// Implementation of the framegrabber configureAcquisition interface
118 /**
119 *
120 * \returns 0 on success
121 * \returns -1 on error
122 */
124
125 /// Implementation of the frameGrabber fps interface
126 /** Just returns the value of m_fps
127 */
128 float fps();
129
130 /// Implementation of the framegrabber startAcquisition interface
131 /**
132 *
133 * \returns 0 on success
134 * \returns -1 on error
135 */
136 int startAcquisition();
137
138 /// Implementation of the framegrabber acquireAndCheckValid interface
139 /**
140 *
141 * \returns 0 on success
142 * \returns -1 on error
143 */
145
146 /// Implementation of the framegrabber loadImageIntoStream interface
147 /**
148 *
149 * \returns 0 on success
150 * \returns -1 on error
151 */
152 int loadImageIntoStream( void *dest /**< [in] */ );
153
154 /// Implementation of the framegrabber reconfig interface
155 /**
156 * \returns 0 on success
157 * \returns -1 on error
158 */
159 int reconfig();
160
161 ///@}
162
163 /** \name Telemeter Interface
164 *
165 * @{
166 */
167 int checkRecordTimes();
168
169 int recordTelem( const telem_fgtimings * );
170};
171
172mcp3008Ctrl::mcp3008Ctrl() : MagAOXApp( MAGAOX_CURRENT_SHA1, MAGAOX_REPO_MODIFIED )
173{
174 return;
175}
176
178{
181
182 config.add( "accel.numChannels",
183 "",
184 "accel.numChannels",
185 argType::Required,
186 "accel",
187 "numChannels",
188 false,
189 "int",
190 "Setting the number of channels needed to readout accelerometers" );
191
192}
193
194int mcp3008Ctrl::loadConfigImpl( mx::app::appConfigurator &_config )
195{
196
199
200 _config( m_numChannels, "fitter.fpsSource" ); // making number of mcp3008 channels we read out configurable
201
202 return 0;
203}
204
206{
207 loadConfigImpl( config );
208}
209
211{
212
215
216 m_adc.connect();
217
218 // INDI prop for user to set fps
219 CREATE_REG_INDI_NEW_NUMBERF( m_indiP_fps, "fps", 0, 10000, 1, "%d", "", "" );
220 m_indiP_fps["current"].setValue( m_fps );
221 m_indiP_fps["target"].setValue( m_fps );
222
223 return 0;
224}
225
227{
230
231 return 0;
232}
233
241
243{
244 /** \todo @PARKER do anything needed to setup the MPC3008*/
245
246 m_values.resize( m_numChannels );
247
249 m_height = 1;
251
252 return 0;
253}
254
256{
257 return m_fps;
258}
259
261{
262 /** \todo @PARKER Do anything needed to start the MPC3008 reading out ... probably nothing*/
263
264 m_time_start = std::chrono::high_resolution_clock::now();
265
266 return 0;
267}
268
270{
271 /** \todo @PARKER Put the loop to wait for the time needed to get an average FPS here*/
272 /* fill */
273
274 while( !m_shutdown && !m_reconfig )
275 {
276 // Get current time
277 auto now = std::chrono::high_resolution_clock::now();
278 auto elapsed = std::chrono::duration_cast<std::chrono::microseconds>( now - m_time_start );
279
280 // Read every 500 microseconds
281 if( elapsed.count() >= m_trigger ) /** \todo @PARKER make m_trigger adjust */
282 {
283 m_time_start = now; // Reset start time
284
285 for( int i = 0; i < m_numChannels; ++i )
286 {
287 m_values[i] = m_adc.read( i ); /** \todo @PARKER*/
288 }
289
290 m_trigger = m_trigger - m_gain * (elapsed.count() - 500);
291
292 return 0;
293 }
294 }
295
296 return 0;
297}
298
300{
301 memcpy( dest, m_values.data(), m_values.size() * sizeof( float ) );
302 return 0;
303}
304
306{
307 return 0;
308}
309
314
316{
317 return recordFGTimings(true);
318}
319
320// Testing for user to select star number
321INDI_NEWCALLBACK_DEFN( mcp3008Ctrl, m_indiP_fps )( const pcf::IndiProperty &ipRecv )
322{
323 if( ipRecv.getName() != m_indiP_fps.getName() )
324 {
325 log<software_error>( { __FILE__, __LINE__, "wrong INDI property received." } );
326 return -1;
327 }
328
329 float target;
330
331 if( indiTargetUpdate( m_indiP_fps, target, ipRecv, true ) < 0 )
332 {
333 log<software_error>( { __FILE__, __LINE__ } );
334 return -1;
335 }
336
337 m_fps = target;
338 m_trigger = m_fps; // Update trigger value based off new fps
339
340 log<text_log>( "set fps = " + std::to_string( m_fps ), logPrio::LOG_NOTICE );
341 return 0;
342}
343
344
345} // namespace app
346} // namespace MagAOX
347
348#endif // mcp3008Ctrl_hpp
unsigned short read(const std::uint8_t channel, const Mode m=Mode::SINGLE) const
Definition MCP3008.cpp:65
The base-class for MagAO-X applications.
Definition MagAOXApp.hpp:73
int m_shutdown
Flag to signal it's time to shutdown. When not 0, the main loop exits.
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
uint32_t m_width
The width of the image, once deinterlaced etc.
uint8_t m_dataType
The ImageStreamIO type code.
bool m_reconfig
Flag to set if a camera reconfiguration requires a framegrabber reset.
uint32_t m_height
The height of the image, once deinterlaced etc.
The MagAO-X MCP3008 Controller.
MCP3008Lib::MCP3008 m_adc
virtual int appLogic()
Implementation of the FSM for mcp3008Ctrl.
int loadConfigImpl(mx::app::appConfigurator &_config)
Implementation of loadConfig logic, separated for testing.
friend class mcp3008Ctrl_test
mcp3008Ctrl()
Default c'tor.
INDI_NEWCALLBACK_DECL(mcp3008Ctrl, m_indiP_fps)
virtual int appShutdown()
Shutdown the app.
static constexpr bool c_frameGrabber_flippable
int acquireAndCheckValid()
Implementation of the framegrabber acquireAndCheckValid interface.
dev::telemeter< mcp3008Ctrl > telemeterT
std::vector< float > m_values
The values read out from the chip.
dev::frameGrabber< mcp3008Ctrl > frameGrabberT
int recordTelem(const telem_fgtimings *)
float m_trigger
The trigger time to readout. Adjusts to match desired FPS.
int loadImageIntoStream(void *dest)
Implementation of the framegrabber loadImageIntoStream interface.
int startAcquisition()
Implementation of the framegrabber startAcquisition interface.
MCP3008Lib::MCP3008 adc
std::chrono::time_point< std::chrono::high_resolution_clock > m_time_start
pcf::IndiProperty m_indiP_fps
virtual int appStartup()
Startup function.
float m_fps
The target FPS.
~mcp3008Ctrl() noexcept
D'tor, declared and defined for noexcept.
int configureAcquisition()
Implementation of the framegrabber configureAcquisition interface.
int reconfig()
Implementation of the framegrabber reconfig interface.
float fps()
Implementation of the frameGrabber fps interface.
int m_numChannels
The number of channels being read out.
#define FRAMEGRABBER_SETUP_CONFIG(cfig)
Call frameGrabberT::setupConfig with error checking for frameGrabber.
#define FRAMEGRABBER_APP_LOGIC
Call frameGrabberT::appLogic with error checking for frameGrabber.
#define FRAMEGRABBER_APP_SHUTDOWN
Call frameGrabberT::appShutdown with error checking for frameGrabber.
#define FRAMEGRABBER_LOAD_CONFIG(cfig)
Call frameGrabberT::loadConfig with error checking for frameGrabber.
#define FRAMEGRABBER_APP_STARTUP
Call frameGrabberT::appStartup with error checking for frameGrabber.
#define INDI_NEWCALLBACK_DEFN(class, prop)
Define the callback for a new property request.
#define CREATE_REG_INDI_NEW_NUMBERF(prop, name, min, max, step, format, label, group)
Create and register a NEW INDI property as a standard number as float, using the standard callback na...
const pcf::IndiProperty & ipRecv
Definition dm.hpp:24
static constexpr logPrioT LOG_NOTICE
A normal but significant condition.
A device base class which saves telemetry.
Definition telemeter.hpp:69
Log entry recording framegrabber timings.
#define TELEMETER_APP_LOGIC
Call telemeter::appLogic with error checking.
#define TELEMETER_LOAD_CONFIG(cfig)
Call telemeter::loadConfig with error checking.
#define TELEMETER_APP_STARTUP
Call telemeter::appStartup with error checking.
#define TELEMETER_SETUP_CONFIG(cfig)
Call telemeter::setupConfig with error checking.
#define TELEMETER_APP_SHUTDOWN
Call telemeter::appShutdown with error checking.