Skip to content
Snippets Groups Projects
Select Git revision
  • 6ac8a5dbb5b3c9f3d320678e18d9fdceb1991ad2
  • master default protected
  • release/202005
  • release/202001
  • release/201912
  • release/201911
  • release/releaseWindowsTestOne
  • release/windowsReleaseTest
  • release/releaseTest
  • release/releaseWindowsTest
  • release/201910
  • release/qt/201910
  • release/windows-test/201910
  • release/201908
  • release/201906
  • release/201905
  • release/201904
  • release/201903
  • release/201902
  • release/201901
  • release/201812
  • 4.0.0
  • 2.2.0
  • 2.1.0
  • 2.0.1
  • 2.0.0
  • 1.4.1
  • 1.4.0
  • 1.3.0
  • 1.2.0
  • 1.1.0
31 results

libav_utils.cpp

Blame
    • Guillaume Roguez's avatar
      6ac8a5db
      daemon: (video) replace anonymous namespace by static keyword · 6ac8a5db
      Guillaume Roguez authored
      Also minor code cleanup.
      
      Rationale:
      The C++ standard has un-deprecated the use of the static keyword.
      But the superiority of anonymous namespaces is mainly for type 
      declarations (including structures and classes).
      
      For variables and functions, the static keyword is functionally
      equivalent to using an anonymous namespace.
      
      But there is a major drawback with the latter!
      Name mangling for anonymous namespaces is such that it's awkward
      to type/find a local symbol declared in them when using a debugger
      such as gdb.
      
      Refs #50143
      
      Change-Id: I925af61d14db08fafa37bfbfeedde95feadd0948
      6ac8a5db
      History
      daemon: (video) replace anonymous namespace by static keyword
      Guillaume Roguez authored
      Also minor code cleanup.
      
      Rationale:
      The C++ standard has un-deprecated the use of the static keyword.
      But the superiority of anonymous namespaces is mainly for type 
      declarations (including structures and classes).
      
      For variables and functions, the static keyword is functionally
      equivalent to using an anonymous namespace.
      
      But there is a major drawback with the latter!
      Name mangling for anonymous namespaces is such that it's awkward
      to type/find a local symbol declared in them when using a debugger
      such as gdb.
      
      Refs #50143
      
      Change-Id: I925af61d14db08fafa37bfbfeedde95feadd0948
    Code owners
    Assign users and groups as approvers for specific file changes. Learn more.
    winmain.cpp 5.69 KiB
    /*
     *  Copyright (C) 2004-2019 Savoir-faire Linux Inc.
     *
     *  Author: Edric Milaret <edric.ladent-milaret@savoirfairelinux.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 3 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.
     */
    
    #include <iostream>
    #include <thread>
    #include <cstring>
    #include <signal.h>
    #include <getopt.h>
    #include <string>
    
    #include "dring.h"
    #include "callmanager_interface.h"
    #include "configurationmanager_interface.h"
    #include "presencemanager_interface.h"
    #ifdef RING_VIDEO
    #include "videomanager_interface.h"
    #endif
    #include "fileutils.h"
    
    #include <windows.h>
    
    using namespace std::placeholders;
    
    bool isActive = false;
    static int ringFlags = 0;
    bool loop = true;
    
    static void
    print_title()
    {
        std::cout
            << "Jami Daemon " << DRing::version()
            << ", by Savoir-faire Linux 2004-2018" << std::endl
            << "https://jami.net/" << std::endl
    #ifdef RING_VIDEO
            << "[Video support enabled]" << std::endl
    #endif
            << std::endl;
    }
    
    static void
    print_usage()
    {
        std::cout << std::endl <<
        "-c, --console \t- Log in console (instead of syslog)" << std::endl <<
        "-d, --debug \t- Debug mode (more verbose)" << std::endl <<
        "-p, --persistent \t- Stay alive after client quits" << std::endl <<
        "--auto-answer \t- Force automatic answer to incoming calls" << std::endl <<
        "-h, --help \t- Print help" << std::endl;
    }
    
    // Parse command line arguments, setting debug options or printing a help
    // message accordingly.
    // returns true if we should quit (i.e. help was printed), false otherwise
    static bool
    parse_args(int argc, char *argv[], bool& persistent)
    {
        int consoleFlag = false;
        int debugFlag = false;
        int helpFlag = false;
        int versionFlag = false;
        int autoAnswer = false;
    
        const struct option long_options[] = {
            /* These options set a flag. */
            {"debug", no_argument, NULL, 'd'},
            {"console", no_argument, NULL, 'c'},
            {"persistent", no_argument, NULL, 'p'},
            {"help", no_argument, NULL, 'h'},
            {"version", no_argument, NULL, 'v'},
            {"auto-answer", no_argument, &autoAnswer, true},
            {0, 0, 0, 0} /* Sentinel */
        };
    
        while (true) {
            /* getopt_long stores the option index here. */
            int option_index = 0;
    
            auto c = getopt_long(argc, argv, "dcphv", long_options, &option_index);
    
            // end of the options
            if (c == -1)
                break;
    
            switch (c) {
                case 'd':
                    debugFlag = true;
                    break;
    
                case 'c':
                    consoleFlag = true;
                    break;
    
                case 'p':
                    persistent = true;
                    break;
    
                case 'h':
                case '?':
                    helpFlag = true;
                    break;
    
                case 'v':
                    versionFlag = true;
                    break;
    
                default:
                    break;
            }
        }
    
        if (helpFlag) {
            print_usage();
            return true;
        }
    
        if (versionFlag) {
            // We've always print the title/version, so we can just exit
            return true;
        }
    
        if (consoleFlag)
            ringFlags |= DRing::DRING_FLAG_CONSOLE_LOG;
    
        if (debugFlag)
            ringFlags |= DRing::DRING_FLAG_DEBUG;
    
        if (autoAnswer)
            ringFlags |= DRing::DRING_FLAG_AUTOANSWER;
    
        return false;
    }
    
    void
    IncomingCall(const std::string& accountId,
        const std::string& callId, const std::string& message)
    {
        (void) accountId;
        (void) message;
        if (not isActive) {
            DRing::accept(callId);
            isActive = true;
        } else
            DRing::refuse(callId);
    }
    
    static int
    run()
    {
        using SharedCallback = std::shared_ptr<DRing::CallbackWrapperBase>;
    
        DRing::init(static_cast<DRing::InitFlag>(ringFlags));
    
        std::map<std::string, SharedCallback> callHandlers;
        callHandlers.insert(DRing::exportable_callback<DRing::CallSignal::IncomingCall>
            (std::bind(&IncomingCall, _1, _2, _3)));
    
        registerSignalHandlers(callHandlers);
    
        if (!DRing::start())
            return -1;
    
        while (loop) {
            Sleep(1000); // milliseconds
        }
    
        DRing::fini();
    
        return 0;
    }
    
    static void
    interrupt()
    {
        loop = false;
    }
    
    static void
    signal_handler(int code)
    {
        std::cerr << "Caught signal " << code
                  << ", terminating..." << std::endl;
        // Unset signal handlers
        signal(SIGINT, SIG_DFL);
        signal(SIGTERM, SIG_DFL);
    
        interrupt();
    }
    
    int
    main(int argc, char *argv [])
    {
        // make a copy as we don't want to modify argv[0], copy it to a vector to
        // guarantee that memory is correctly managed/exception safe
        std::string programName {argv[0]};
        std::vector<char> writable(programName.size() + 1);
        std::copy(std::begin(programName), std::end(programName),std::begin(writable));
    
        ring::fileutils::set_program_dir(writable.data());
    
        print_title();
    
        bool persistent = false;
        if (parse_args(argc, argv, persistent))
            return 0;
    
        // TODO: Block signals for all threads but the main thread, decide how/if we should
        // handle other signals
        signal(SIGINT, signal_handler);
        signal(SIGTERM, signal_handler);
    
        return run();
    }