// NasalPositioned_cppbind.cxx -- expose FGPositioned classes to Nasal // // Port of NasalPositioned.cpp to the new nasal/cppbind helpers. Will replace // old NasalPositioned.cpp once finished. // // Copyright (C) 2013 Thomas Geymayer // // 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. #ifdef HAVE_CONFIG_H # include "config.h" #endif #include "NasalPositioned.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include
#include #include #include #include typedef nasal::Ghost NasalPositioned; typedef nasal::Ghost NasalRunway; typedef nasal::Ghost NasalParking; typedef nasal::Ghost NasalAirport; typedef nasal::Ghost NasalCommStation; typedef nasal::Ghost NasalNavRecord; typedef nasal::Ghost NasalRunway; typedef nasal::Ghost NasalFix; //------------------------------------------------------------------------------ naRef to_nasal_helper(naContext c, flightgear::SID* sid) { // TODO SID ghost return nasal::to_nasal(c, sid->ident()); } //------------------------------------------------------------------------------ naRef to_nasal_helper(naContext c, flightgear::STAR* star) { // TODO STAR ghost return nasal::to_nasal(c, star->ident()); } //------------------------------------------------------------------------------ naRef to_nasal_helper(naContext c, flightgear::Approach* iap) { // TODO Approach ghost return nasal::to_nasal(c, iap->ident()); } //------------------------------------------------------------------------------ static naRef f_navaid_course(FGNavRecord& nav, naContext) { if( !( nav.type() == FGPositioned::ILS || nav.type() == FGPositioned::LOC ) ) return naNil(); double radial = nav.get_multiuse(); return naNum(SGMiscd::normalizePeriodic(0.5, 360.5, radial)); } //------------------------------------------------------------------------------ static FGRunwayBaseRef f_airport_runway(FGAirport& apt, std::string ident) { boost::to_upper(ident); if( apt.hasRunwayWithIdent(ident) ) return apt.getRunwayByIdent(ident); else if( apt.hasHelipadWithIdent(ident) ) return apt.getHelipadByIdent(ident); return 0; } //------------------------------------------------------------------------------ template std::vector extract( const std::vector& in, T (C2::*getter)() const ) { std::vector ret(in.size()); std::transform(in.begin(), in.end(), ret.begin(), std::mem_fun(getter)); return ret; } //------------------------------------------------------------------------------ static naRef f_airport_comms(FGAirport& apt, const nasal::CallContext& ctx) { FGPositioned::Type comm_type = FGPositioned::typeFromName( ctx.getArg(0) ); // if we have an explicit type, return a simple vector of frequencies if( comm_type != FGPositioned::INVALID ) return ctx.to_nasal ( extract( apt.commStationsOfType(comm_type), &flightgear::CommStation::freqMHz ) ); else // otherwise return a vector of ghosts, one for each comm station. return ctx.to_nasal(apt.commStations()); } //------------------------------------------------------------------------------ FGRunwayRef runwayFromNasalArg( const FGAirport& apt, const nasal::CallContext& ctx, size_t index = 0 ) { if( index >= ctx.argc ) return FGRunwayRef(); try { std::string ident = ctx.getArg(index); if( !ident.empty() ) { if( !apt.hasRunwayWithIdent(ident) ) // TODO warning/exception? return FGRunwayRef(); return apt.getRunwayByIdent(ident); } } catch(...) {} // TODO warn/error if no runway? return ctx.from_nasal(ctx.args[index]); } //------------------------------------------------------------------------------ static naRef f_airport_sids(FGAirport& apt, const nasal::CallContext& ctx) { FGRunway* rwy = runwayFromNasalArg(apt, ctx); return ctx.to_nasal ( extract(rwy ? rwy->getSIDs() : apt.getSIDs(), &flightgear::SID::ident) ); } //------------------------------------------------------------------------------ static naRef f_airport_stars(FGAirport& apt, const nasal::CallContext& ctx) { FGRunway* rwy = runwayFromNasalArg(apt, ctx); return ctx.to_nasal ( extract(rwy ? rwy->getSTARs() : apt.getSTARs(), &flightgear::STAR::ident) ); } //------------------------------------------------------------------------------ static naRef f_airport_approaches(FGAirport& apt, const nasal::CallContext& ctx) { FGRunway* rwy = runwayFromNasalArg(apt, ctx); flightgear::ProcedureType type = flightgear::PROCEDURE_INVALID; std::string type_str = ctx.getArg(1); if( !type_str.empty() ) { boost::to_upper(type_str); if( type_str == "NDB" ) type = flightgear::PROCEDURE_APPROACH_NDB; else if( type_str == "VOR" ) type = flightgear::PROCEDURE_APPROACH_VOR; else if( type_str == "ILS" ) type = flightgear::PROCEDURE_APPROACH_ILS; else if( type_str == "RNAV") type = flightgear::PROCEDURE_APPROACH_RNAV; } return ctx.to_nasal ( extract( rwy ? rwy->getApproaches(type) // no runway specified, report them all : apt.getApproaches(type), &flightgear::Approach::ident ) ); } //------------------------------------------------------------------------------ static FGParkingList f_airport_parking(FGAirport& apt, nasal::CallContext ctx) { std::string type = ctx.getArg(0); bool only_available = ctx.getArg(1); FGAirportDynamics* dynamics = apt.getDynamics(); PositionedIDVec parkings = flightgear::NavDataCache::instance() ->airportItemsOfType(apt.guid(), FGPositioned::PARKING); FGParkingList ret; BOOST_FOREACH(PositionedID parking, parkings) { // filter out based on availability and type if( only_available && !dynamics->isParkingAvailable(parking) ) continue; FGParking* park = dynamics->getParking(parking); if( !type.empty() && (park->getType() != type) ) continue; ret.push_back(park); } return ret; } /** * Extract a SGGeod from a nasal function argument list. * * , * {"lat": , "lon": } * geo.Coord.new() (aka. {"_lat": , "_lon": }) */ static bool extractGeod(nasal::CallContext& ctx, SGGeod& result) { if( !ctx.argc ) return false; if( ctx.isGhost(0) ) { FGPositionedRef pos = ctx.from_nasal(ctx.requireArg(0)); if( pos ) { result = pos->geod(); ctx.popFront(); return true; } } else if( ctx.isHash(0) ) { nasal::Hash pos_hash = ctx.requireArg(0); // check for manual latitude / longitude names naRef lat = pos_hash.get("lat"), lon = pos_hash.get("lon"); if( naIsNum(lat) && naIsNum(lon) ) { result = SGGeod::fromDeg( ctx.from_nasal(lon), ctx.from_nasal(lat) ); ctx.popFront(); return true; } // geo.Coord uses _lat/_lon in radians // TODO should we check if its really a geo.Coord? lat = pos_hash.get("_lat"); lon = pos_hash.get("_lon"); if( naIsNum(lat) && naIsNum(lon) ) { result = SGGeod::fromRad( ctx.from_nasal(lon), ctx.from_nasal(lat) ); ctx.popFront(); return true; } } else if( ctx.isNumeric(0) && ctx.isNumeric(1) ) { // lat, lon result = SGGeod::fromDeg( ctx.requireArg(1), ctx.requireArg(0) ); ctx.popFront(2); return true; } return false; } /** * Extract position from ctx or return current aircraft position if not given. */ static SGGeod getPosition(nasal::CallContext& ctx) { SGGeod pos; if( !extractGeod(ctx, pos) ) pos = globals->get_aircraft_position(); return pos; } //------------------------------------------------------------------------------ // Returns Nasal ghost for particular or nearest airport of a , or nil // on error. // // airportinfo(); e.g. "KSFO" // airportinfo(); type := ("airport"|"seaport"|"heliport") // airportinfo() same as airportinfo("airport") // airportinfo(, [, ]); static naRef f_airportinfo(nasal::CallContext ctx) { SGGeod pos = getPosition(ctx); if( ctx.argc > 1 ) naRuntimeError(ctx.c, "airportinfo() with invalid function arguments"); // optional type/ident std::string ident("airport"); if( ctx.isString(0) ) ident = ctx.requireArg(0); FGAirport::TypeRunwayFilter filter; if( !filter.fromTypeString(ident) ) // user provided an , hopefully return ctx.to_nasal(FGAirport::findByIdent(ident)); double maxRange = 10000.0; // expose this? or pick a smaller value? return ctx.to_nasal( FGAirport::findClosest(pos, maxRange, &filter) ); } /** * findAirportsWithinRange([,] [, type]) */ static naRef f_findAirportsWithinRange(nasal::CallContext ctx) { SGGeod pos = getPosition(ctx); double range_nm = ctx.requireArg(0); FGAirport::TypeRunwayFilter filter; // defaults to airports only filter.fromTypeString( ctx.getArg(1) ); FGPositionedList apts = FGPositioned::findWithinRange(pos, range_nm, &filter); FGPositioned::sortByRange(apts, pos); return ctx.to_nasal(apts); } /** * findAirportsByICAO( [, type]) */ static naRef f_findAirportsByICAO(nasal::CallContext ctx) { std::string prefix = ctx.requireArg(0); FGAirport::TypeRunwayFilter filter; // defaults to airports only filter.fromTypeString( ctx.getArg(1) ); return ctx.to_nasal( FGPositioned::findAllWithIdent(prefix, &filter, false) ); } // Returns vector of data hash for navaid of a , nil on error // navaids sorted by ascending distance // navinfo([,],[],[]) // lat/lon (numeric): use latitude/longitude instead of ac position // type: ("fix"|"vor"|"ndb"|"ils"|"dme"|"tacan"|"any") // id: (partial) id of the fix // examples: // navinfo("vor") returns all vors // navinfo("HAM") return all navaids who's name start with "HAM" // navinfo("vor", "HAM") return all vor who's name start with "HAM" //navinfo(34,48,"vor","HAM") return all vor who's name start with "HAM" // sorted by distance relative to lat=34, lon=48 static naRef f_navinfo(nasal::CallContext ctx) { SGGeod pos = getPosition(ctx); std::string id = ctx.getArg(0); FGNavList::TypeFilter filter; if( filter.fromTypeString(id) ) id = ctx.getArg(1); else if( ctx.argc > 1 ) naRuntimeError(ctx.c, "navinfo() already got an ident"); return ctx.to_nasal( FGNavList::findByIdentAndFreq(pos, id, 0.0, &filter) ); } //------------------------------------------------------------------------------ static naRef f_findWithinRange(nasal::CallContext ctx) { SGGeod pos = getPosition(ctx); double range_nm = ctx.requireArg(0); std::string typeSpec = ctx.getArg(1); FGPositioned::TypeFilter filter(FGPositioned::TypeFilter::fromString(typeSpec)); FGPositionedList items = FGPositioned::findWithinRange(pos, range_nm, &filter); FGPositioned::sortByRange(items, pos); return ctx.to_nasal(items); } static naRef f_findByIdent(nasal::CallContext ctx) { std::string prefix = ctx.requireArg(0); std::string typeSpec = ctx.getArg(1); FGPositioned::TypeFilter filter(FGPositioned::TypeFilter::fromString(typeSpec)); bool exact = ctx.getArg(2, false); return ctx.to_nasal( FGPositioned::findAllWithIdent(prefix, &filter, exact) ); } static naRef f_findByName(nasal::CallContext ctx) { std::string prefix = ctx.requireArg(0); std::string typeSpec = ctx.getArg(1); FGPositioned::TypeFilter filter(FGPositioned::TypeFilter::fromString(typeSpec)); return ctx.to_nasal( FGPositioned::findAllWithName(prefix, &filter, false) ); } //------------------------------------------------------------------------------ static naRef f_courseAndDistance(nasal::CallContext ctx) { SGGeod from = globals->get_aircraft_position(), to, pos; bool ok = extractGeod(ctx, pos); if (!ok) { naRuntimeError(ctx.c, "invalid arguments to courseAndDistance"); } if (extractGeod(ctx, to)) { from = pos; // we parsed both FROM and TO args, so first was FROM } else { to = pos; // only parsed one arg, so FROM is current } double course, course2, d; SGGeodesy::inverse(from, to, course, course2, d); naRef result = naNewVector(ctx.c); naVec_append(result, naNum(course)); naVec_append(result, naNum(d * SG_METER_TO_NM)); return result; } static naRef f_sortByRange(nasal::CallContext ctx) { FGPositionedList items = ctx.requireArg(0); ctx.popFront(); FGPositioned::sortByRange(items, getPosition(ctx)); return ctx.to_nasal(items); } //------------------------------------------------------------------------------ // Get difference between two lists of positioned objects. // // For every element in old_list not in new_list the callback cb_remove is // called with the removed element as single argument. For every element in // new_list not in old_list cb_add is called. // // diff(old_list, new_list, cb_add[, cb_remove]) // // example: // # Print all fixes within a distance of 320 to 640 miles // diff( findWithinRange(320, "fix"), // findWithinRange(640, "fix"), // func(p) print('found fix: ', p.id) ); static naRef f_diff(nasal::CallContext ctx) { typedef simgear::ListDiff Diff; Diff::List old_list = ctx.requireArg(0), new_list = ctx.requireArg(1); Diff::Callback cb_add = ctx.requireArg(2), cb_rm = ctx.getArg(3); // Note that FGPositionedRef instances are only compared for pointer equality. // As the NavCache caches every queried positioned instance it is guaranteed // that only one instance of every positioned object can exist. Therefore we // can make the comparison faster by just comparing pointers and not also the // guid. // (On my machine the difference is 0.27s vs 0.17s) Diff::inplace(old_list, new_list, cb_add, cb_rm); return naNil(); } //------------------------------------------------------------------------------ naRef initNasalPositioned_cppbind(naRef globalsRef, naContext c) { NasalPositioned::init("Positioned") .member("id", &FGPositioned::ident) .member("ident", &FGPositioned::ident) // TODO to we really need id and ident? .member("name", &FGPositioned::name) .member("type", &FGPositioned::typeString) .member("lat", &FGPositioned::latitude) .member("lon", &FGPositioned::longitude) .member("elevation", &FGPositioned::elevationM); NasalRunway::init("Runway") .bases(); NasalParking::init("Parking") .bases(); NasalCommStation::init("CommStation") .bases() .member("frequency", &flightgear::CommStation::freqMHz); NasalNavRecord::init("Navaid") .bases() .member("frequency", &FGNavRecord::get_freq) .member("range_nm", &FGNavRecord::get_range) .member("course", &f_navaid_course); NasalFix::init("Fix") .bases(); NasalAirport::init("FGAirport") .bases() .member("has_metar", &FGAirport::getMetar) .member("runways", &FGAirport::getRunwayMap) .member("helipads", &FGAirport::getHelipadMap) .member("taxiways", &FGAirport::getTaxiways) .member("pavements", &FGAirport::getPavements) .method("runway", &f_airport_runway) .method("helipad", &f_airport_runway) .method("tower", &FGAirport::getTowerLocation) .method("comms", &f_airport_comms) .method("sids", &f_airport_sids) .method("stars", &f_airport_stars) .method("getApproachList", f_airport_approaches) .method("parking", &f_airport_parking) .method("getSid", &FGAirport::findSIDWithIdent) .method("getStar", &FGAirport::findSTARWithIdent) .method("getIAP", &FGAirport::findApproachWithIdent) .method("tostring", &FGAirport::toString); nasal::Hash globals(globalsRef, c), positioned( globals.createHash("positioned") ); positioned.set("airportinfo", &f_airportinfo); positioned.set("findAirportsWithinRange", f_findAirportsWithinRange); positioned.set("findAirportsByICAO", &f_findAirportsByICAO); positioned.set("navinfo", &f_navinfo); positioned.set("findWithinRange", &f_findWithinRange); positioned.set("findByIdent", &f_findByIdent); positioned.set("findByName", &f_findByName); positioned.set("courseAndDistance", &f_courseAndDistance); positioned.set("sortByRange", &f_sortByRange); positioned.set("diff", &f_diff); return naNil(); }