From e0cb122abe1ed6f9ad868ffe0eda5feb2d64ab0e Mon Sep 17 00:00:00 2001 From: Maximilian Wuttke Date: Sun, 14 Dec 2014 09:50:18 +0100 Subject: [PATCH 1/5] Merge branch 'filter_audio' of https://github.com/mwuttke97/qTox into filter_audio Added some `#ifdef QTOX_FILTER_AUDIO`s. Conflicts: qtox.pro Conflicts: src/coreav.cpp --- install_libfilteraudio.sh | 31 +++++++++++++++ qtox.pro | 6 +++ src/audiofilterer.cpp | 52 ++++++++++++++++++++++++++ src/audiofilterer.h | 41 ++++++++++++++++++++ src/core.h | 6 +++ src/coreav.cpp | 44 +++++++++++++++++++++- src/misc/settings.cpp | 10 +++++ src/misc/settings.h | 4 ++ src/widget/form/settings/avform.cpp | 11 ++++++ src/widget/form/settings/avform.h | 1 + src/widget/form/settings/avsettings.ui | 11 +++++- 11 files changed, 213 insertions(+), 4 deletions(-) create mode 100644 install_libfilteraudio.sh create mode 100644 src/audiofilterer.cpp create mode 100644 src/audiofilterer.h diff --git a/install_libfilteraudio.sh b/install_libfilteraudio.sh new file mode 100644 index 000000000..d0f5bdd38 --- /dev/null +++ b/install_libfilteraudio.sh @@ -0,0 +1,31 @@ +#!/bin/sh +CURRENT_DIR=$DIRSTACK +SOURCE_DIR="filter_audio/" +SOURCE_PATH="./../" +LIB_DIR="/usr/local/lib/" +INCLUDE_DIR="/usr/local/include/" + +echo "Clone filter_audio from GitHub.com" +cd $SOURCE_PATH +git clone --quiet https://github.com/irungentoo/filter_audio.git $SOURCE_DIR + +echo "Compile filter_audio" +cd $SOURCE_DIR +gcc -c -fPIC filter_audio.c aec/*.c agc/*.c ns/*.c other/*.c -lm -lpthread + +echo "Create shared object file" +gcc *.o -shared -o libfilteraudio.so + +echo "Clean up" +rm *.o + +echo "Install libfilteraudio.so" +sudo cp libfilteraudio.so $LIB_DIR + +echo "Install include files" +sudo cp *.h $INCLUDE_DIR + +echo "Finished." +cd $CURRENT_DIR +exit 1 + diff --git a/qtox.pro b/qtox.pro index cbd4f3dc1..9dc7722c0 100644 --- a/qtox.pro +++ b/qtox.pro @@ -234,6 +234,12 @@ SOURCES += \ src/widget/form/settings/advancedform.cpp \ src/audio.cpp +contains(DEFINES, QTOX_FILTER_AUDIO) { + HEADERS += src/audiofilterer.h + SOURCES += src/audiofilterer.cpp + unix|win32: LIBS += -lfilteraudio +} + contains(DEFINES, QTOX_PLATFORM_EXT) { HEADERS += src/platform/timer.h SOURCES += src/platform/timer_osx.cpp \ diff --git a/src/audiofilterer.cpp b/src/audiofilterer.cpp new file mode 100644 index 000000000..f88ae875b --- /dev/null +++ b/src/audiofilterer.cpp @@ -0,0 +1,52 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program is libre 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 COPYING file for more details. +*/ + + +#ifdef QTOX_FILTER_AUDIO + +#include "audiofilterer.h" +extern "C"{ +#include +} + +void AudioFilterer::startFilter(unsigned int fs) +{ + closeFilter(); + filter = new_filter_audio(fs); +} + +void AudioFilterer::closeFilter() +{ + if (filter) + kill_filter_audio(filter); +} + + +void AudioFilterer::filterAudio(int16_t* data, int framesize) +{ + if (!filter) + return; + + filter_audio(filter, (int16_t*) data, framesize); +} + + +AudioFilterer::~AudioFilterer() +{ + closeFilter(); +} + +#endif // QTOX_FILTER_AUDIO diff --git a/src/audiofilterer.h b/src/audiofilterer.h new file mode 100644 index 000000000..3a61d4e9f --- /dev/null +++ b/src/audiofilterer.h @@ -0,0 +1,41 @@ +/* + Copyright (C) 2014 by Project Tox + + This file is part of qTox, a Qt-based graphical interface for Tox. + + This program is libre 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 COPYING file for more details. +*/ + +#ifdef QTOX_FILTER_AUDIO +#ifndef AUDIOFILTERER_H +#define AUDIOFILTERER_H +#include + +#ifndef _FILTER_AUDIO +typedef struct Filter_Audio Filter_Audio; +#endif + +class AudioFilterer +{ +public: + explicit AudioFilterer() = default; + ~AudioFilterer(); + + void startFilter(unsigned int fs); + void filterAudio(int16_t* data, int framesize); + void closeFilter(); + +private: + struct Filter_Audio* filter{nullptr}; +}; + +#endif // AUDIOFILTERER_H +#endif // QTOX_FILTER_AUDIO diff --git a/src/core.h b/src/core.h index 2f323bca2..ffa4ba0c9 100644 --- a/src/core.h +++ b/src/core.h @@ -33,6 +33,9 @@ class QTimer; class QString; class CString; class VideoSource; +#ifdef QTOX_FILTER_AUDIO +class AudioFilterer; +#endif class Core : public QObject { @@ -283,6 +286,9 @@ private: int dhtServerId; static QList fileSendQueue, fileRecvQueue; static ToxCall calls[TOXAV_MAX_CALLS]; +#ifdef QTOX_FILTER_AUDIO + static AudioFilterer * filterer[TOXAV_MAX_CALLS]; +#endif static QHash groupCalls; // Maps group IDs to ToxGroupCalls QMutex fileSendMutex, messageSendMutex; bool ready; diff --git a/src/coreav.cpp b/src/coreav.cpp index 13f6dd8c0..41d0fe490 100644 --- a/src/coreav.cpp +++ b/src/coreav.cpp @@ -17,10 +17,17 @@ #include "core.h" #include "video/camera.h" #include "audio.h" +#ifdef QTOX_FILTER_AUDIO +#include "audiofilterer.h" +#endif +#include "misc/settings.h" #include #include ToxCall Core::calls[TOXAV_MAX_CALLS]; +#ifdef QTOX_FILTER_AUDIO +AudioFilterer * Core::filterer[TOXAV_MAX_CALLS] { nullptr}; +#endif const int Core::videobufsize{TOXAV_MAX_VIDEO_WIDTH * TOXAV_MAX_VIDEO_HEIGHT * 4}; uint8_t* Core::videobuf; @@ -65,6 +72,20 @@ void Core::prepareCall(int friendId, int callId, ToxAv* toxav, bool videoEnabled calls[callId].sendVideoTimer->start(); Camera::getInstance()->subscribe(); } + +#ifdef QTOX_FILTER_AUDIO + if (Settings::getInstance().getFilterAudio()) + { + Core::filterer[callId] = new AudioFilterer(); + filterer[callId]->startFilter(48000); + } + else + { + if (filterer[callId]) + delete filterer[callId]; + filterer[callId] = nullptr; + } +#endif } void Core::onAvMediaChange(void* toxav, int32_t callId, void* core) @@ -246,8 +267,16 @@ void Core::sendCallAudio(int callId, ToxAv* toxav) return; } +#ifdef QTOX_FILTER_AUDIO + if (filterer[callId]) + { + filterer[callId]->filterAudio((int16_t*) buf, framesize); + } +#endif if ((r = toxav_send_audio(toxav, callId, dest, r)) < 0) + { qDebug() << "Core: toxav_send_audio error"; + } } calls[callId].sendAudioTimer->start(); } @@ -294,14 +323,16 @@ void Core::sendCallVideo(int callId) void Core::micMuteToggle(int callId) { - if (calls[callId].active) { + if (calls[callId].active) + { calls[callId].muteMic = !calls[callId].muteMic; } } void Core::volMuteToggle(int callId) { - if (calls[callId].active) { + if (calls[callId].active) + { calls[callId].muteVol = !calls[callId].muteVol; alSourcef(calls[callId].alSource, AL_GAIN, calls[callId].muteVol ? 0.f : 1.f); } @@ -321,6 +352,15 @@ void Core::onAvCancel(void* _toxav, int32_t callId, void* core) calls[callId].active = false; +#ifdef QTOX_FILTER_AUDIO + if (filterer[callId]) + { + filterer[callId]->closeFilter(); + delete filterer[callId]; + filterer[callId] = nullptr; + } +#endif + emit static_cast(core)->avCancel(friendId, callId); } diff --git a/src/misc/settings.cpp b/src/misc/settings.cpp index b14fe2e34..5ec16448a 100644 --- a/src/misc/settings.cpp +++ b/src/misc/settings.cpp @@ -198,6 +198,7 @@ void Settings::load() s.beginGroup("Audio"); inDev = s.value("inDev", "").toString(); outDev = s.value("outDev", "").toString(); + filterAudio = s.value("filterAudio", false).toBool(); s.endGroup(); // Read the embedded DHT bootsrap nodes list if needed @@ -340,6 +341,7 @@ void Settings::save(QString path, bool writeFriends) s.beginGroup("Audio"); s.setValue("inDev", inDev); s.setValue("outDev", outDev); + s.setValue("filterAudio", filterAudio); s.endGroup(); if (!writeFriends || currentProfile.isEmpty()) // Core::switchConfiguration @@ -898,6 +900,14 @@ void Settings::setOutDev(const QString& deviceSpecifier) outDev = deviceSpecifier; } +bool Settings::getFilterAudio() const{ + return filterAudio; +} + +void Settings::setFilterAudio(bool newValue){ + filterAudio = newValue; +} + QString Settings::getFriendAdress(const QString &publicKey) const { QString key = ToxID::fromString(publicKey).publicKey; diff --git a/src/misc/settings.h b/src/misc/settings.h index 848ade407..64ff6009e 100644 --- a/src/misc/settings.h +++ b/src/misc/settings.h @@ -128,6 +128,9 @@ public: QString getOutDev() const; void setOutDev(const QString& deviceSpecifier); + bool getFilterAudio() const; + void setFilterAudio(bool newValue); + // Assume all widgets have unique names // Don't use it to save every single thing you want to save, use it // for some general purpose widgets, such as MainWindows or Splitters, @@ -298,6 +301,7 @@ private: // Audio QString inDev; QString outDev; + bool filterAudio; struct friendProp { diff --git a/src/widget/form/settings/avform.cpp b/src/widget/form/settings/avform.cpp index a506b287a..903fc7e6a 100644 --- a/src/widget/form/settings/avform.cpp +++ b/src/widget/form/settings/avform.cpp @@ -33,12 +33,19 @@ AVForm::AVForm() : bodyUI = new Ui::AVSettings; bodyUI->setupUi(this); +#ifdef QTOX_FILTER_AUDIO + bodyUI->filterAudio->setChecked(Settings::getInstance().getFilterAudio()); +#else + bodyUI->filterAudio->setDisabled(true); +#endif + connect(Camera::getInstance(), &Camera::propProbingFinished, this, &AVForm::onPropProbingFinished); connect(Camera::getInstance(), &Camera::resolutionProbingFinished, this, &AVForm::onResProbingFinished); auto qcomboboxIndexChanged = (void(QComboBox::*)(const QString&)) &QComboBox::currentIndexChanged; connect(bodyUI->inDevCombobox, qcomboboxIndexChanged, this, &AVForm::onInDevChanged); connect(bodyUI->outDevCombobox, qcomboboxIndexChanged, this, &AVForm::onOutDevChanged); + connect(bodyUI->filterAudio, SIGNAL(toggled(bool)), this, SLOT(onFilterAudioToggled(bool))); connect(bodyUI->rescanButton, &QPushButton::clicked, this, [=](){getAudioInDevices(); getAudioOutDevices();}); } @@ -189,3 +196,7 @@ void AVForm::onOutDevChanged(const QString& deviceDescriptor) Settings::getInstance().setOutDev(deviceDescriptor); Audio::openOutput(deviceDescriptor); } + +void AVForm::onFilterAudioToggled(bool filterAudio){ + Settings::getInstance().setFilterAudio(filterAudio); +} diff --git a/src/widget/form/settings/avform.h b/src/widget/form/settings/avform.h index 2d1e873ad..64b34d410 100644 --- a/src/widget/form/settings/avform.h +++ b/src/widget/form/settings/avform.h @@ -52,6 +52,7 @@ private slots: // audio void onInDevChanged(const QString& deviceDescriptor); void onOutDevChanged(const QString& deviceDescriptor); + void onFilterAudioToggled(bool filterAudio); // camera void onPropProbingFinished(Camera::Prop prop, double val); diff --git a/src/widget/form/settings/avsettings.ui b/src/widget/form/settings/avsettings.ui index f7e1afc8d..e46c7d43f 100644 --- a/src/widget/form/settings/avsettings.ui +++ b/src/widget/form/settings/avsettings.ui @@ -30,8 +30,8 @@ 0 0 - 810 - 496 + 808 + 618 @@ -96,6 +96,13 @@ + + + + Filter audio + + + From 8934a11dbffce6b0611d97d5748668a5cdb8ba8a Mon Sep 17 00:00:00 2001 From: Sean Qureshi Date: Tue, 16 Dec 2014 02:51:13 -0800 Subject: [PATCH 2/5] Squashed 'osx/updater/' content from commit a558402 git-subtree-dir: osx/updater git-subtree-split: a5584021435fc74a1e5da447524b37dcf7311bbc --- .gitignore | 23 +++++ README.md | 13 +++ qtox_sudo.m | 273 ++++++++++++++++++++++++++++++++++++++++++++++++++++ updater.go | 133 +++++++++++++++++++++++++ 4 files changed, 442 insertions(+) create mode 100644 .gitignore create mode 100644 README.md create mode 100644 qtox_sudo.m create mode 100644 updater.go diff --git a/.gitignore b/.gitignore new file mode 100644 index 000000000..836562412 --- /dev/null +++ b/.gitignore @@ -0,0 +1,23 @@ +# Compiled Object files, Static and Dynamic libs (Shared Objects) +*.o +*.a +*.so + +# Folders +_obj +_test + +# Architecture specific extensions/prefixes +*.[568vq] +[568vq].out + +*.cgo1.go +*.cgo2.c +_cgo_defun.c +_cgo_gotypes.go +_cgo_export.* + +_testmain.go + +*.exe +*.test diff --git a/README.md b/README.md new file mode 100644 index 000000000..66d65555f --- /dev/null +++ b/README.md @@ -0,0 +1,13 @@ +The qTox OS X updater is a mix of objective C and Go compiled as static binaries used do effortless updates in the background. + +It uses Objective C to access Apples own security framework and call some long dead APIs in order to give the statically linked go updater permissions to install the latest build without prompting the user for every file. + +* Release commits: ``https://github.com/Tox/qTox_updater`` +* Development commits: ``https://github.mit.edu/sean-2/updater`` + +Compiling: + +* ```clang qtox_sudo.m -framework corefoundation -framework security -framework cocoa -Os -o qtox_sudo``` +* ```go build updater.go``` + +(Starting with this commit all commits will be signed with [this key](http://pgp.mit.edu/pks/lookup?op=get&search=0x13D2043169D25DF4).) diff --git a/qtox_sudo.m b/qtox_sudo.m new file mode 100644 index 000000000..f9de6bcc0 --- /dev/null +++ b/qtox_sudo.m @@ -0,0 +1,273 @@ +// Modifications listed here GPLv3: https://gist.githubusercontent.com/stqism/2e82352026915f8f6ab3/raw/fca6f6f16fb8d61a64b6053dacf50c3433c2e0af/gistfile1.txt +// +// Copyright 2009 Performant Design, LLC. All rights reserved. +// Copyright (C) 2014 Tox Foundation +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// +// 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, see . +// + +#import + +#include +#include +#include +#include + +char *addFileToPath(const char *path, const char *filename) { + char *outbuf; + char *lc; + + lc = (char *)path + strlen(path) - 1; + + if (lc < path || *lc != '/') { + lc = NULL; + } + + while (*filename == '/') { + filename++; + } + + outbuf = malloc(strlen(path) + strlen(filename) + 1 + (lc == NULL ? 1 : 0)); + + sprintf(outbuf, "%s%s%s", path, (lc == NULL) ? "/" : "", filename); + + return outbuf; +} + +int isExecFile(const char *name) { + struct stat s; + + return (!access(name, X_OK) && !stat(name, &s) && S_ISREG(s.st_mode)); +} + +char *which(const char *filename) +{ + char *path, *p, *n; + + path = getenv("PATH"); + + if (!path) { + return NULL; + } + + p = path = strdup(path); + + while (p) { + n = strchr(p, ':'); + + if (n) { + *n++ = '\0'; + } + + if (*p != '\0') { + p = addFileToPath(p, filename); + + if (isExecFile(p)) { + free(path); + + return p; + } + + free(p); + } + + p = n; + } + + free(path); + + return NULL; +} + +int cocoaSudo(char *executable, char *commandArgs[], char *icon, char *prompt) { + int retVal = 1; + OSStatus status; + AuthorizationRef authRef; + + AuthorizationItem right = {kAuthorizationRightExecute, 0, NULL, 0}; + AuthorizationRights rightSet = {1, &right}; + + AuthorizationEnvironment myAuthorizationEnvironment; + AuthorizationItem kAuthEnv[2]; + myAuthorizationEnvironment.items = kAuthEnv; + + AuthorizationFlags flags = kAuthorizationFlagDefaults; + + + if (prompt && icon) { + kAuthEnv[0].name = kAuthorizationEnvironmentPrompt; + kAuthEnv[0].valueLength = strlen(prompt); + kAuthEnv[0].value = prompt; + kAuthEnv[0].flags = 0; + + kAuthEnv[1].name = kAuthorizationEnvironmentIcon; + kAuthEnv[1].valueLength = strlen(icon); + kAuthEnv[1].value = icon; + kAuthEnv[1].flags = 0; + + myAuthorizationEnvironment.count = 2; + } + else if (prompt) { + kAuthEnv[0].name = kAuthorizationEnvironmentPrompt; + kAuthEnv[0].valueLength = strlen(prompt); + kAuthEnv[0].value = prompt; + kAuthEnv[0].flags = 0; + + myAuthorizationEnvironment.count = 1; + } + else if (icon) { + kAuthEnv[0].name = kAuthorizationEnvironmentIcon; + kAuthEnv[0].valueLength = strlen(icon); + kAuthEnv[0].value = icon; + kAuthEnv[0].flags = 0; + + myAuthorizationEnvironment.count = 1; + } + else { + myAuthorizationEnvironment.count = 0; + } + + status = AuthorizationCreate(NULL, &myAuthorizationEnvironment, flags, &authRef); + + if (status != errAuthorizationSuccess) { + NSLog(@"Could not create authorization reference object."); + status = errAuthorizationBadAddress; + } + else { + flags = kAuthorizationFlagDefaults | + kAuthorizationFlagInteractionAllowed | + kAuthorizationFlagPreAuthorize | + kAuthorizationFlagExtendRights; + + status = AuthorizationCopyRights(authRef, &rightSet, &myAuthorizationEnvironment, flags, NULL); + } + + if (status == errAuthorizationSuccess) { + FILE *ioPipe; + char buffer[1024]; + int bytesRead; + + flags = kAuthorizationFlagDefaults; + status = AuthorizationExecuteWithPrivileges(authRef, executable, flags, commandArgs, &ioPipe); + + /* Just pipe processes' stdout to our stdout for now; hopefully can add stdin pipe later as well */ + + for (;;) { + bytesRead = fread(buffer, sizeof(char), 1024, ioPipe); + + if (bytesRead < 1) { + break; + } + + write(STDOUT_FILENO, buffer, bytesRead * sizeof(char)); + } + + pid_t pid; + int pidStatus; + + do { + pid = wait(&pidStatus); + } + while (pid != -1); + + if (status == errAuthorizationSuccess) { + retVal = 0; + } + } + else { + AuthorizationFree(authRef, kAuthorizationFlagDestroyRights); + authRef = NULL; + + if (status != errAuthorizationCanceled) { + // pre-auth failed + NSLog(@"Pre-auth failed"); + } + } + + return retVal; +} + +void usage(char *appNameFull) { + char *appName = strrchr(appNameFull, '/'); + + if (appName == NULL) { + appName = appNameFull; + } + else { + appName++; + } + + fprintf(stderr, "usage: %s command\n", appName); +} + +int main(int argc, char *argv[]) { + NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init]; + + int retVal = 1; + int programArgsStartAt = 1; + char *icon = NULL; + char *prompt = NULL; + + if (programArgsStartAt >= argc) { + usage(argv[0]); + } + else { + char *executable; + + if (strchr(argv[programArgsStartAt], '/')) { + executable = isExecFile(argv[programArgsStartAt]) ? strdup(argv[programArgsStartAt]) : NULL; + } + else { + executable = which(argv[programArgsStartAt]); + } + + if (executable) { + char **commandArgs = malloc((argc - programArgsStartAt) * sizeof(char**)); + + memcpy(commandArgs, argv + programArgsStartAt + 1, (argc - programArgsStartAt - 1) * sizeof(char**)); + + commandArgs[argc - programArgsStartAt - 1] = NULL; + + retVal = cocoaSudo(executable, commandArgs, icon, prompt); + + free(commandArgs); + free(executable); + } + else { + fprintf(stderr, "Unable to find %s\n", argv[programArgsStartAt]); + + usage(argv[0]); + } + } + + if (prompt) { + free(prompt); + } + + [pool release]; + + return retVal; +} diff --git a/updater.go b/updater.go new file mode 100644 index 000000000..ae7b898d6 --- /dev/null +++ b/updater.go @@ -0,0 +1,133 @@ +package main + +import ( + "fmt" + "io/ioutil" + "log" + "os" + "os/exec" + "os/user" + "syscall" + + "bitbucket.org/kardianos/osext" +) + +var custom_user string + +func fs_type(path string) int { + //name := "FileOrDir" + f, err := os.Open(path) + if err != nil { + fmt.Println(err) + return -1 + } + defer f.Close() + fi, err := f.Stat() + if err != nil { + fmt.Println(err) + return -1 + } + switch mode := fi.Mode(); { + case mode.IsDir(): + return 0 + case mode.IsRegular(): + return 1 + } + + return -1 +} + +func install(path string, pathlen int) int { + files, _ := ioutil.ReadDir(path) + + for _, file := range files { + if fs_type(path+file.Name()) == 1 { + + addpath := "" + if len(path) != pathlen { + addpath = path[pathlen:len(path)] + } + + fmt.Print("Installing: ") + fmt.Println("/Applications/qtox.app/Contents/" + addpath + file.Name()) + if _, err := os.Stat("/Applications/qtox.app/Contents/" + file.Name()); os.IsNotExist(err) { + newfile := exec.Command("/usr/libexec/authopen", "-c", "-x", "-m", "drwxrwxr-x+", "/Applications/qtox.app/Contents/"+addpath+file.Name()) + newfile.Run() + } + + cat := exec.Command("/bin/cat", path+file.Name()) + + auth := exec.Command("/usr/libexec/authopen", "-w", "/Applications/qtox.app/Contents/"+addpath+file.Name()) + auth.Stdin, _ = cat.StdoutPipe() + auth.Stdout = os.Stdout + auth.Stderr = os.Stderr + _ = auth.Start() + _ = cat.Run() + _ = auth.Wait() + + } else { + install(path+file.Name()+"/", pathlen) + } + } + return 0 +} + +func main() { + syscall.Setuid(0) + usr, e := user.Current() + if e != nil { + log.Fatal(e) + } + +CHECK: + if usr.Name != "System Administrator" { + fmt.Println("Not running as root, relaunching") + + appdir, _ := osext.Executable() + appdir_len := len(appdir) + sudo_path := appdir[0:(appdir_len-7)] + "qtox_sudo" //qtox_sudo is a fork of cocoasudo with all of its flags and other features stripped out + + if _, err := os.Stat(sudo_path); os.IsNotExist(err) { + fmt.Println("Error: No qtox_sudo binary installed, falling back") + custom_user = usr.Name + usr.Name = "System Administrator" + goto CHECK + } + + relaunch := exec.Command(sudo_path, appdir, usr.Name) + relaunch.Stdout = os.Stdout + relaunch.Stderr = os.Stderr + relaunch.Run() + return + + } else { + + if len(os.Args) > 1 || custom_user != "" { + + if custom_user == "" { + custom_user = os.Args[1] + } + + update_dir := "/Users/" + custom_user + "/Library/Preferences/tox/update/" + if _, err := os.Stat(update_dir); os.IsNotExist(err) { + fmt.Println("Error: No update folder, is check for updates enabled?") + return + } + fmt.Println("qTox Updater") + + killqtox := exec.Command("/usr/bin/killall", "qtox") + _ = killqtox.Run() + + install(update_dir, len(update_dir)) + + os.RemoveAll(update_dir) + fmt.Println("Update metadata wiped, launching qTox") + launchqtox := exec.Command("/usr/bin/open", "-b", "im.tox.qtox") + launchqtox.Run() + + } else { + fmt.Println("Error: no user passed") + } + + } +} From e30b74a984e8451f2df0eae85b2801225cb88faf Mon Sep 17 00:00:00 2001 From: Sean Qureshi Date: Tue, 16 Dec 2014 02:51:31 -0800 Subject: [PATCH 3/5] Switch to subtree --- osx/updater.go | 92 -------------------------------------------------- 1 file changed, 92 deletions(-) delete mode 100644 osx/updater.go diff --git a/osx/updater.go b/osx/updater.go deleted file mode 100644 index b063a251d..000000000 --- a/osx/updater.go +++ /dev/null @@ -1,92 +0,0 @@ -package main - -import ( - "fmt" - "io/ioutil" - "log" - "os" - "os/exec" - "os/user" -) - -func fs_type(path string) int { - //name := "FileOrDir" - f, err := os.Open(path) - if err != nil { - fmt.Println(err) - return -1 - } - defer f.Close() - fi, err := f.Stat() - if err != nil { - fmt.Println(err) - return -1 - } - switch mode := fi.Mode(); { - case mode.IsDir(): - return 0 - case mode.IsRegular(): - return 1 - } - - return -1 -} - -func install(path string, pathlen int) int { - files, _ := ioutil.ReadDir(path) - - for _, file := range files { - if fs_type(path+file.Name()) == 1 { - - addpath := "" - if len(path) != pathlen { - addpath = path[pathlen:len(path)] - } - - fmt.Print("Installing: ") - fmt.Println("/Applications/qtox.app/Contents/" + addpath + file.Name()) - if _, err := os.Stat("/Applications/qtox.app/Contents/" + file.Name()); os.IsNotExist(err) { - newfile := exec.Command("/usr/libexec/authopen", "-c", "-x", "-m", "drwxrwxr-x+", "/Applications/qtox.app/Contents/"+addpath+file.Name()) - newfile.Run() - } - - cat := exec.Command("/bin/cat", path+file.Name()) - - auth := exec.Command("/usr/libexec/authopen", "-w", "/Applications/qtox.app/Contents/"+addpath+file.Name()) - auth.Stdin, _ = cat.StdoutPipe() - auth.Stdout = os.Stdout - auth.Stderr = os.Stderr - _ = auth.Start() - _ = cat.Run() - _ = auth.Wait() - - } else { - install(path+file.Name()+"/", pathlen) - } - } - return 0 -} - -func main() { - usr, e := user.Current() - if e != nil { - log.Fatal(e) - } - - update_dir := usr.HomeDir + "/Library/Preferences/tox/update/" - if _, err := os.Stat(update_dir); os.IsNotExist(err) { - fmt.Println("Error: No update folder, is check for updates enabled?") - return - } - fmt.Println("qTox Updater") - - killqtox := exec.Command("/usr/bin/killall", "qtox") - _ = killqtox.Run() - - install(update_dir, len(update_dir)) - - os.RemoveAll(update_dir) - fmt.Println("Update metadata wiped, launching qTox") - launchqtox := exec.Command("/usr/bin/open", "-b", "im.tox.qtox") - launchqtox.Run() -} From fc749880ebaf259689fa26550c3332043dcb1971 Mon Sep 17 00:00:00 2001 From: Dubslow Date: Tue, 16 Dec 2014 18:05:13 -0600 Subject: [PATCH 4/5] minor style tweaks --- src/audiofilterer.cpp | 1 + src/coreav.cpp | 3 +-- src/misc/settings.cpp | 6 ++++-- src/widget/form/settings/avform.cpp | 3 ++- 4 files changed, 8 insertions(+), 5 deletions(-) diff --git a/src/audiofilterer.cpp b/src/audiofilterer.cpp index f88ae875b..2253a6f60 100644 --- a/src/audiofilterer.cpp +++ b/src/audiofilterer.cpp @@ -32,6 +32,7 @@ void AudioFilterer::closeFilter() { if (filter) kill_filter_audio(filter); + filter = nullptr; } diff --git a/src/coreav.cpp b/src/coreav.cpp index 41d0fe490..5fb12e128 100644 --- a/src/coreav.cpp +++ b/src/coreav.cpp @@ -81,8 +81,7 @@ void Core::prepareCall(int friendId, int callId, ToxAv* toxav, bool videoEnabled } else { - if (filterer[callId]) - delete filterer[callId]; + delete filterer[callId]; filterer[callId] = nullptr; } #endif diff --git a/src/misc/settings.cpp b/src/misc/settings.cpp index 5ec16448a..ad2c9fbb3 100644 --- a/src/misc/settings.cpp +++ b/src/misc/settings.cpp @@ -900,11 +900,13 @@ void Settings::setOutDev(const QString& deviceSpecifier) outDev = deviceSpecifier; } -bool Settings::getFilterAudio() const{ +bool Settings::getFilterAudio() const +{ return filterAudio; } -void Settings::setFilterAudio(bool newValue){ +void Settings::setFilterAudio(bool newValue) +{ filterAudio = newValue; } diff --git a/src/widget/form/settings/avform.cpp b/src/widget/form/settings/avform.cpp index 903fc7e6a..31f9b6dd9 100644 --- a/src/widget/form/settings/avform.cpp +++ b/src/widget/form/settings/avform.cpp @@ -197,6 +197,7 @@ void AVForm::onOutDevChanged(const QString& deviceDescriptor) Audio::openOutput(deviceDescriptor); } -void AVForm::onFilterAudioToggled(bool filterAudio){ +void AVForm::onFilterAudioToggled(bool filterAudio) +{ Settings::getInstance().setFilterAudio(filterAudio); } From 5cbe175ddc490a6b2b2b68288e30913cfa7f4cb0 Mon Sep 17 00:00:00 2001 From: Dubslow Date: Tue, 16 Dec 2014 18:50:28 -0600 Subject: [PATCH 5/5] update the build system --- bootstrap.sh | 14 +++++++++++- install_libfilteraudio.sh | 45 +++++++++++++++++++++++---------------- qtox.pro | 18 ++++++++++++++-- 3 files changed, 56 insertions(+), 21 deletions(-) mode change 100644 => 100755 install_libfilteraudio.sh diff --git a/bootstrap.sh b/bootstrap.sh index 3188d9ecc..d705a02c0 100755 --- a/bootstrap.sh +++ b/bootstrap.sh @@ -15,6 +15,7 @@ SODIUM_VER=1.0.0 # directory names of cloned repositories SODIUM_DIR=libsodium-$SODIUM_VER TOX_CORE_DIR=libtoxcore-latest +FILTER_AUDIO_DIR=filter_audio # this boolean describes whether the installation of # libsodium should be skipped or not @@ -42,6 +43,11 @@ if [ -z "$TOX_CORE_DIR" ]; then exit 1 fi +if [ -z "$FILTER_AUDIO_DIR" ]; then + echo "internal error detected!" + echo "FILTER_AUDIO_DIR should not be empty... aborting" + exit 1 +fi ########## check input parameters ########## @@ -95,7 +101,7 @@ mkdir -p ${BASE_DIR} # if exists, otherwise cloning them may fail rm -rf ${BASE_DIR}/${SODIUM_DIR} rm -rf ${BASE_DIR}/${TOX_CORE_DIR} - +rm -rf ${BASE_DIR}/${FILTER_AUDIO_DIR} ############### install step ############### @@ -122,6 +128,12 @@ if [[ $TOX_ONLY = "false" ]]; then fi popd + + if [[ $GLOBAL = "false" ]]; then + ./install_libfilteraudio.sh ${BASE_DIR}/${FILTER_AUDIO_DIR} ${BASE_DIR} + else + ./install_libfilteraudio.sh ${BASE_DIR}/${FILTER_AUDIO_DIR} + fi fi # clone current master of libtoxcore diff --git a/install_libfilteraudio.sh b/install_libfilteraudio.sh old mode 100644 new mode 100755 index d0f5bdd38..010bcbbac --- a/install_libfilteraudio.sh +++ b/install_libfilteraudio.sh @@ -1,31 +1,40 @@ #!/bin/sh -CURRENT_DIR=$DIRSTACK -SOURCE_DIR="filter_audio/" -SOURCE_PATH="./../" -LIB_DIR="/usr/local/lib/" -INCLUDE_DIR="/usr/local/include/" -echo "Clone filter_audio from GitHub.com" -cd $SOURCE_PATH -git clone --quiet https://github.com/irungentoo/filter_audio.git $SOURCE_DIR +if [ -z $1 ]; then + SOURCE_DIR="filter_audio/" +else + SOURCE_DIR="$1/" +fi -echo "Compile filter_audio" +if [ -z "$2" ]; then + LIB_DIR="/usr/local/lib/" + INCLUDE_DIR="/usr/local/include/" +else + LIB_DIR="$2/lib/" + INCLUDE_DIR="$2/include/" +fi + +echo "Cloning filter_audio from GitHub.com" +git clone https://github.com/irungentoo/filter_audio.git $SOURCE_DIR + +echo "Compiling filter_audio" cd $SOURCE_DIR gcc -c -fPIC filter_audio.c aec/*.c agc/*.c ns/*.c other/*.c -lm -lpthread -echo "Create shared object file" +echo "Creating shared object file" gcc *.o -shared -o libfilteraudio.so -echo "Clean up" +echo "Cleaning up" rm *.o -echo "Install libfilteraudio.so" -sudo cp libfilteraudio.so $LIB_DIR +muhcmd="cp libfilteraudio.so $LIB_DIR" +[ -z "$2" ] && muhcmd="sudo $muhcmd" +echo "Installing libfilteraudio.so with $muhcmd" +$muhcmd -echo "Install include files" -sudo cp *.h $INCLUDE_DIR +muhcmd="cp *.h $INCLUDE_DIR" +[ -z "$2" ] && muhcmd="sudo $muhcmd" +echo "Installing include files with $muhcmd" +$muhcmd echo "Finished." -cd $CURRENT_DIR -exit 1 - diff --git a/qtox.pro b/qtox.pro index 9dc7722c0..46be9d92d 100644 --- a/qtox.pro +++ b/qtox.pro @@ -59,6 +59,12 @@ contains(DISABLE_PLATFORM_EXT, YES) { DEFINES += QTOX_PLATFORM_EXT } +contains(DISABLE_FILTER_AUDIO, YES) { + +} else { + DEFINES += QTOX_FILTER_AUDIO +} + contains(JENKINS,YES) { INCLUDEPATH += ./libs/include/ } else { @@ -78,6 +84,7 @@ win32 { QMAKE_INFO_PLIST = osx/info.plist LIBS += -L$$PWD/libs/lib/ -ltoxcore -ltoxav -ltoxencryptsave -ltoxdns -lsodium -lvpx -framework OpenAL -lopencv_core -lopencv_highgui contains(DEFINES, QTOX_PLATFORM_EXT) { LIBS += -framework IOKit -framework CoreFoundation } + contains(DEFINES, QTOX_FILTER_AUDIO) { } } else { # If we're building a package, static link libtox[core,av] and libsodium, since they are not provided by any package contains(STATICPKG, YES) { @@ -94,8 +101,16 @@ win32 { LIBS += -lX11 -lXss } + contains(DEFINES, QTOX_FILTER_AUDIO) { + contains(STATICPKG, YES) { + LIBS += -Wl,-Bstatic -lfilteraudio + } else { + LIBS += -lfilteraudio + } + } + contains(JENKINS, YES) { - LIBS = ./libs/lib/libtoxav.a ./libs/lib/libvpx.a ./libs/lib/libopus.a ./libs/lib/libtoxdns.a ./libs/lib/libtoxencryptsave.a ./libs/lib/libtoxcore.a ./libs/lib/libsodium.a /usr/lib/libopencv_core.so /usr/lib/libopencv_highgui.so /usr/lib/libopencv_imgproc.so -lopenal -lX11 -lXss -s + LIBS = ./libs/lib/libtoxav.a ./libs/lib/libvpx.a ./libs/lib/libopus.a ./libs/lib/libtoxdns.a ./libs/lib/libtoxencryptsave.a ./libs/lib/libtoxcore.a ./libs/lib/libsodium.a ./libs/lib/libfilteraudio.a /usr/lib/libopencv_core.so /usr/lib/libopencv_highgui.so /usr/lib/libopencv_imgproc.so -lopenal -lX11 -lXss -s } } } @@ -237,7 +252,6 @@ SOURCES += \ contains(DEFINES, QTOX_FILTER_AUDIO) { HEADERS += src/audiofilterer.h SOURCES += src/audiofilterer.cpp - unix|win32: LIBS += -lfilteraudio } contains(DEFINES, QTOX_PLATFORM_EXT) {