// environment-ctrl.hxx -- controller for environment information. // // Written by David Megginson, started May 2002. // // Copyright (C) 2002 David Megginson - david@megginson.com // // This program is free software; you can redistribute it and/or // modify it under the terms of the GNU General Public License as // published by the Free Software Foundation; either version 2 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, but // WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software // Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. // // $Id$ #ifndef _ENVIRONMENT_CTRL_HXX #define _ENVIRONMENT_CTRL_HXX #include #include #include #if defined(ENABLE_THREADS) && ENABLE_THREADS # include # include #endif #ifdef SG_HAVE_STD_INCLUDES # include #else # include #endif #include #include SG_USING_STD(queue); SG_USING_STD(vector); class SGPropertyNode; #include "environment.hxx" #include "fgmetar.hxx" /** * Interface to control environment information for a specific location. */ class FGEnvironmentCtrl : public SGSubsystem { public: FGEnvironmentCtrl (); virtual ~FGEnvironmentCtrl (); virtual void setEnvironment (FGEnvironment * environment); virtual const FGEnvironment * getEnvironment () const { return _environment; } virtual void setLongitudeDeg (double lon_deg); virtual void setLatitudeDeg (double lat_deg); virtual void setElevationFt (double elev_ft); virtual void setPosition (double lon_deg, double lat_deg, double elev_ft); virtual double getLongitudeDeg () const { return _lon_deg; } virtual double getLatitudeDeg () const { return _lat_deg; } virtual double getElevationFt () const { return _elev_ft; } protected: FGEnvironment * _environment; double _lon_deg; double _lat_deg; double _elev_ft; }; /** * Environment controller using user-supplied parameters. */ class FGUserDefEnvironmentCtrl : public FGEnvironmentCtrl { public: FGUserDefEnvironmentCtrl (); virtual ~FGUserDefEnvironmentCtrl (); virtual void init (); virtual void update (double dt); private: SGPropertyNode * _base_wind_speed_node; SGPropertyNode * _gust_wind_speed_node; double _current_wind_speed_kt; double _delta_wind_speed_kt; }; /** * Interplation controller using user-supplied parameters. */ class FGInterpolateEnvironmentCtrl : public FGEnvironmentCtrl { public: FGInterpolateEnvironmentCtrl (); virtual ~FGInterpolateEnvironmentCtrl (); virtual void init (); virtual void reinit (); virtual void update (double delta_time_sec); private: struct bucket { double altitude_ft; FGEnvironment environment; bool operator< (const bucket &b) const; }; void read_table (const SGPropertyNode * node, vector &table); void do_interpolate (vector &table, double altitude_ft, FGEnvironment * environment); FGEnvironment env1, env2; // temporaries vector _boundary_table; vector _aloft_table; }; // A convenience wrapper around FGMetar struct FGMetarResult { string icao; FGMetar *m; }; /** * Interplation controller using the FGMetar class */ class FGMetarEnvironmentCtrl : public FGEnvironmentCtrl { public: FGMetarEnvironmentCtrl (); virtual ~FGMetarEnvironmentCtrl (); virtual void init (); virtual void reinit (); virtual void update (double delta_time_sec); virtual void setEnvironment (FGEnvironment * environment); private: FGInterpolateEnvironmentCtrl *env; string _icao; float station_elevation_ft; float search_interval_sec; float same_station_interval_sec; float search_elapsed; float fetch_elapsed; FGAirport last_apt; SGPropertyNode *proxy_host; SGPropertyNode *proxy_port; SGPropertyNode *proxy_auth; SGPropertyNode *metar_max_age; FGMetarResult fetch_data( const string &icao ); virtual void update_metar_properties( const FGMetar *m ); void update_env_config(); private: #if defined(ENABLE_THREADS) && ENABLE_THREADS /** * FIFO queue which holds a pointer to the fetched metar data. */ SGBlockingQueue < string > request_queue; /** * FIFO queue which holds a pointer to the fetched metar data. */ SGBlockingQueue < FGMetarResult > result_queue; #else /** * FIFO queue which holds a pointer to the fetched metar data. */ queue < string > request_queue; /** * FIFO queue which holds a pointer to the fetched metar data. */ queue < FGMetarResult > result_queue; #endif #if defined(ENABLE_THREADS) && ENABLE_THREADS /** * This class represents the thread of execution responsible for * fetching the metar data. */ class MetarThread : public SGThread { public: MetarThread( FGMetarEnvironmentCtrl* f ) : fetcher(f) {} ~MetarThread() {} /** * Fetche the metar data from the NOAA. */ void run(); private: FGMetarEnvironmentCtrl *fetcher; private: // not implemented. MetarThread(); MetarThread( const MetarThread& ); MetarThread& operator=( const MetarThread& ); }; friend class MetarThread; /** * Metar data fetching thread. */ MetarThread* thread; /** * Lock and synchronize access to metar queue. */ SGMutex mutex; SGPthreadCond metar_cond; /** * Thread cleanup handler. */ friend void metar_cleanup_handler( void* ); #endif // ENABLE_THREADS int _error_count; double _dt; double _error_dt; }; #endif // _ENVIRONMENT_CTRL_HXX