diff options
Diffstat (limited to 'src/frontend/qt_sdl')
46 files changed, 12156 insertions, 0 deletions
| diff --git a/src/frontend/qt_sdl/AudioSettingsDialog.cpp b/src/frontend/qt_sdl/AudioSettingsDialog.cpp new file mode 100644 index 0000000..2ff5307 --- /dev/null +++ b/src/frontend/qt_sdl/AudioSettingsDialog.cpp @@ -0,0 +1,103 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <stdio.h> +#include <QFileDialog> + +#include "types.h" +#include "Platform.h" +#include "Config.h" +#include "PlatformConfig.h" + +#include "AudioSettingsDialog.h" +#include "ui_AudioSettingsDialog.h" + + +AudioSettingsDialog* AudioSettingsDialog::currentDlg = nullptr; + +extern char* EmuDirectory; + + +AudioSettingsDialog::AudioSettingsDialog(QWidget* parent) : QDialog(parent), ui(new Ui::AudioSettingsDialog) +{ +    ui->setupUi(this); +    setAttribute(Qt::WA_DeleteOnClose); + +    oldVolume = Config::AudioVolume; + +    ui->slVolume->setValue(Config::AudioVolume); + +    grpMicMode = new QButtonGroup(this); +    grpMicMode->addButton(ui->rbMicNone,     0); +    grpMicMode->addButton(ui->rbMicExternal, 1); +    grpMicMode->addButton(ui->rbMicNoise,    2); +    grpMicMode->addButton(ui->rbMicWav,      3); +    connect(grpMicMode, SIGNAL(buttonClicked(int)), this, SLOT(onChangeMicMode(int))); +    grpMicMode->button(Config::MicInputType)->setChecked(true); + +    ui->txtMicWavPath->setText(Config::MicWavPath); + +    bool iswav = (Config::MicInputType == 3); +    ui->txtMicWavPath->setEnabled(iswav); +    ui->btnMicWavBrowse->setEnabled(iswav); +} + +AudioSettingsDialog::~AudioSettingsDialog() +{ +    delete ui; +} + +void AudioSettingsDialog::on_AudioSettingsDialog_accepted() +{ +    Config::MicInputType = grpMicMode->checkedId(); +    strncpy(Config::MicWavPath, ui->txtMicWavPath->text().toStdString().c_str(), 1023); Config::MicWavPath[1023] = '\0'; +    Config::Save(); + +    closeDlg(); +} + +void AudioSettingsDialog::on_AudioSettingsDialog_rejected() +{ +    Config::AudioVolume = oldVolume; + +    closeDlg(); +} + +void AudioSettingsDialog::on_slVolume_valueChanged(int val) +{ +    Config::AudioVolume = val; +} + +void AudioSettingsDialog::onChangeMicMode(int mode) +{ +    bool iswav = (mode == 3); +    ui->txtMicWavPath->setEnabled(iswav); +    ui->btnMicWavBrowse->setEnabled(iswav); +} + +void AudioSettingsDialog::on_btnMicWavBrowse_clicked() +{ +    QString file = QFileDialog::getOpenFileName(this, +                                                "Select WAV file...", +                                                EmuDirectory, +                                                "WAV files (*.wav);;Any file (*.*)"); + +    if (file.isEmpty()) return; + +    ui->txtMicWavPath->setText(file); +} diff --git a/src/frontend/qt_sdl/AudioSettingsDialog.h b/src/frontend/qt_sdl/AudioSettingsDialog.h new file mode 100644 index 0000000..3bafa30 --- /dev/null +++ b/src/frontend/qt_sdl/AudioSettingsDialog.h @@ -0,0 +1,69 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef AUDIOSETTINGSDIALOG_H +#define AUDIOSETTINGSDIALOG_H + +#include <QDialog> +#include <QButtonGroup> + +namespace Ui { class AudioSettingsDialog; } +class AudioSettingsDialog; + +class AudioSettingsDialog : public QDialog +{ +    Q_OBJECT + +public: +    explicit AudioSettingsDialog(QWidget* parent); +    ~AudioSettingsDialog(); + +    static AudioSettingsDialog* currentDlg; +    static AudioSettingsDialog* openDlg(QWidget* parent) +    { +        if (currentDlg) +        { +            currentDlg->activateWindow(); +            return currentDlg; +        } + +        currentDlg = new AudioSettingsDialog(parent); +        currentDlg->show(); +        return currentDlg; +    } +    static void closeDlg() +    { +        currentDlg = nullptr; +    } + +private slots: +    void on_AudioSettingsDialog_accepted(); +    void on_AudioSettingsDialog_rejected(); + +    void on_slVolume_valueChanged(int val); +    void onChangeMicMode(int mode); +    void on_btnMicWavBrowse_clicked(); + +private: +    Ui::AudioSettingsDialog* ui; + +    int oldVolume; +    QButtonGroup* grpMicMode; +}; + +#endif // AUDIOSETTINGSDIALOG_H diff --git a/src/frontend/qt_sdl/AudioSettingsDialog.ui b/src/frontend/qt_sdl/AudioSettingsDialog.ui new file mode 100644 index 0000000..bcaf937 --- /dev/null +++ b/src/frontend/qt_sdl/AudioSettingsDialog.ui @@ -0,0 +1,174 @@ +<?xml version="1.0" encoding="UTF-8"?> +<ui version="4.0"> + <class>AudioSettingsDialog</class> + <widget class="QDialog" name="AudioSettingsDialog"> +  <property name="geometry"> +   <rect> +    <x>0</x> +    <y>0</y> +    <width>482</width> +    <height>230</height> +   </rect> +  </property> +  <property name="sizePolicy"> +   <sizepolicy hsizetype="Fixed" vsizetype="Fixed"> +    <horstretch>0</horstretch> +    <verstretch>0</verstretch> +   </sizepolicy> +  </property> +  <property name="windowTitle"> +   <string>Audio settings - melonDS</string> +  </property> +  <layout class="QVBoxLayout" name="verticalLayout"> +   <property name="sizeConstraint"> +    <enum>QLayout::SetFixedSize</enum> +   </property> +   <item> +    <widget class="QGroupBox" name="groupBox"> +     <property name="title"> +      <string>Audio output</string> +     </property> +     <layout class="QGridLayout" name="gridLayout_2"> +      <item row="0" column="0"> +       <widget class="QLabel" name="label"> +        <property name="text"> +         <string>Volume:</string> +        </property> +       </widget> +      </item> +      <item row="0" column="1"> +       <widget class="QSlider" name="slVolume"> +        <property name="whatsThis"> +         <string><html><head/><body><p>Controls the volume of the audio output.</p></body></html></string> +        </property> +        <property name="maximum"> +         <number>256</number> +        </property> +        <property name="pageStep"> +         <number>16</number> +        </property> +        <property name="orientation"> +         <enum>Qt::Horizontal</enum> +        </property> +       </widget> +      </item> +     </layout> +    </widget> +   </item> +   <item> +    <widget class="QGroupBox" name="groupBox_2"> +     <property name="title"> +      <string>Microphone input</string> +     </property> +     <layout class="QGridLayout" name="gridLayout"> +      <item row="3" column="1"> +       <widget class="QLineEdit" name="txtMicWavPath"> +        <property name="minimumSize"> +         <size> +          <width>290</width> +          <height>0</height> +         </size> +        </property> +        <property name="whatsThis"> +         <string><html><head/><body><p>Forward a WAV file to the emulated microphone.</p><p>This input mode is activated by holding the microphone hotkey (see Input and Hotkeys).</p></body></html></string> +        </property> +       </widget> +      </item> +      <item row="3" column="0"> +       <widget class="QRadioButton" name="rbMicWav"> +        <property name="whatsThis"> +         <string><html><head/><body><p>Forward a WAV file to the emulated microphone.</p><p>This input mode is activated by holding the microphone hotkey (see Input and Hotkeys).</p></body></html></string> +        </property> +        <property name="text"> +         <string>WAV file:</string> +        </property> +       </widget> +      </item> +      <item row="3" column="2"> +       <widget class="QPushButton" name="btnMicWavBrowse"> +        <property name="text"> +         <string>Browse...</string> +        </property> +       </widget> +      </item> +      <item row="1" column="0" colspan="3"> +       <widget class="QRadioButton" name="rbMicExternal"> +        <property name="whatsThis"> +         <string><html><head/><body><p>Input from an external microphone, if available, will be forwarded to the emulated microphone.</p></body></html></string> +        </property> +        <property name="text"> +         <string>External microphone</string> +        </property> +       </widget> +      </item> +      <item row="2" column="0" colspan="3"> +       <widget class="QRadioButton" name="rbMicNoise"> +        <property name="whatsThis"> +         <string><html><head/><body><p>Noise will be forwarded to the emulated microphone, simulating blowing into the microphone.</p><p>This input mode is activated by holding the microphone hotkey (see Input and Hotkeys).</p></body></html></string> +        </property> +        <property name="text"> +         <string>Blow noise</string> +        </property> +       </widget> +      </item> +      <item row="0" column="0" colspan="3"> +       <widget class="QRadioButton" name="rbMicNone"> +        <property name="whatsThis"> +         <string><html><head/><body><p>No microphone input.</p></body></html></string> +        </property> +        <property name="text"> +         <string>None</string> +        </property> +       </widget> +      </item> +     </layout> +    </widget> +   </item> +   <item> +    <widget class="QDialogButtonBox" name="buttonBox"> +     <property name="orientation"> +      <enum>Qt::Horizontal</enum> +     </property> +     <property name="standardButtons"> +      <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set> +     </property> +    </widget> +   </item> +  </layout> + </widget> + <resources/> + <connections> +  <connection> +   <sender>buttonBox</sender> +   <signal>accepted()</signal> +   <receiver>AudioSettingsDialog</receiver> +   <slot>accept()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>248</x> +     <y>254</y> +    </hint> +    <hint type="destinationlabel"> +     <x>157</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> +  <connection> +   <sender>buttonBox</sender> +   <signal>rejected()</signal> +   <receiver>AudioSettingsDialog</receiver> +   <slot>reject()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>316</x> +     <y>260</y> +    </hint> +    <hint type="destinationlabel"> +     <x>286</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> + </connections> +</ui> diff --git a/src/frontend/qt_sdl/CMakeLists.txt b/src/frontend/qt_sdl/CMakeLists.txt new file mode 100644 index 0000000..37a8ff1 --- /dev/null +++ b/src/frontend/qt_sdl/CMakeLists.txt @@ -0,0 +1,114 @@ +project(qt_sdl) + +SET(SOURCES_QT_SDL +	main.cpp +	main_shaders.h +	EmuSettingsDialog.cpp +	InputConfigDialog.cpp +	VideoSettingsDialog.cpp +	AudioSettingsDialog.cpp +	WifiSettingsDialog.cpp +	Input.cpp +	LAN_PCap.cpp +	LAN_Socket.cpp +	OSD.cpp +	OSD_shaders.h +	font.h +	Platform.cpp +	PlatformConfig.cpp +	 +	../Util_ROM.cpp +	../Util_Video.cpp +	../Util_Audio.cpp +	../FrontendUtil.h +	../mic_blow.h +	 +	../../../melon.qrc +) + +if (WIN32) +	set(CMAKE_RC_COMPILE_OBJECT "<CMAKE_RC_COMPILER> -i <SOURCE> -o <OBJECT>") +endif() + +if (BUILD_STATIC AND QT5_STATIC_DIR) +	set(QT5_STATIC_BASE ${QT5_STATIC_DIR}/lib/cmake/Qt5) +	set(Qt5_DIR ${QT5_STATIC_BASE}) +	set(Qt5Core_DIR ${QT5_STATIC_BASE}Core) +	set(Qt5Gui_DIR ${QT5_STATIC_BASE}Gui) +	set(Qt5Widgets_DIR ${QT5_STATIC_BASE}Widgets) +endif() + +find_package(Qt5 COMPONENTS Core Gui Widgets REQUIRED) + +set(CMAKE_AUTOMOC ON) +set(CMAKE_AUTOUIC ON) +set(CMAKE_AUTORCC ON) + +find_package(PkgConfig REQUIRED) +pkg_check_modules(SDL2 REQUIRED sdl2) + +if (WIN32 AND (CMAKE_BUILD_TYPE STREQUAL Release)) +	add_executable(melonDS WIN32 ${SOURCES_QT_SDL}) +else() +	add_executable(melonDS ${SOURCES_QT_SDL}) +endif() + +target_include_directories(melonDS PRIVATE ${SDL2_INCLUDE_DIRS}) +target_include_directories(melonDS PUBLIC "${CMAKE_CURRENT_SOURCE_DIR}") +target_include_directories(melonDS PUBLIC "${CMAKE_CURRENT_SOURCE_DIR}/..") +target_include_directories(melonDS PUBLIC "${CMAKE_CURRENT_SOURCE_DIR}/../..") +target_link_libraries(melonDS core) + +if (BUILD_STATIC) +	target_link_libraries(melonDS -static ${SDL2_LIBRARIES}) +else() +	target_link_libraries(melonDS ${SDL2_LIBRARIES}) +endif() + +if (UNIX) +	option(UNIX_PORTABLE "Make a portable build that looks for its configuration in the current directory" OFF) +	if (UNIX_PORTABLE) +		add_definitions(-DUNIX_PORTABLE) +	endif() + +	find_package(PkgConfig REQUIRED) +	pkg_check_modules(GTK3 REQUIRED gtk+-3.0) + +	target_include_directories(melonDS PRIVATE ${GTK3_INCLUDE_DIRS}) +	target_link_libraries(melonDS ${GTK3_LIBRARIES}) + +	ADD_DEFINITIONS(${GTK3_CFLAGS_OTHER}) + +	add_custom_command(OUTPUT melon_grc.c +		COMMAND glib-compile-resources --sourcedir=${CMAKE_SOURCE_DIR} +				--target=${CMAKE_CURRENT_BINARY_DIR}/melon_grc.c +				--generate-source "${CMAKE_SOURCE_DIR}/melon_grc.xml" +		COMMAND glib-compile-resources --sourcedir=${CMAKE_SOURCE_DIR} +				--target=${CMAKE_CURRENT_BINARY_DIR}/melon_grc.h +				--generate-header "${CMAKE_SOURCE_DIR}/melon_grc.xml") + +	if (CMAKE_SYSTEM_NAME STREQUAL "Linux") +		target_link_libraries(melonDS dl Qt5::Core Qt5::Gui Qt5::Widgets) +	endif () + +	target_sources(melonDS PUBLIC melon_grc.c) +elseif (WIN32) +	target_sources(melonDS PUBLIC "${CMAKE_SOURCE_DIR}/melon.rc") +	 +	target_link_libraries(melonDS comctl32 d2d1 dwrite uxtheme ws2_32 iphlpapi gdi32) +	if (BUILD_STATIC) +		target_link_libraries(melonDS imm32 winmm version setupapi -static Qt5::Core Qt5::Gui Qt5::Widgets z zstd) +	else() +		target_link_libraries(melonDS Qt5::Core Qt5::Gui Qt5::Widgets) +	endif() +endif () + +install(FILES ../../../net.kuribo64.melonDS.desktop DESTINATION ${CMAKE_INSTALL_PREFIX}/share/applications) +install(FILES ../../../icon/melon_16x16.png DESTINATION ${CMAKE_INSTALL_PREFIX}/share/icons/hicolor/16x16/apps RENAME net.kuribo64.melonDS.png) +install(FILES ../../../icon/melon_32x32.png DESTINATION ${CMAKE_INSTALL_PREFIX}/share/icons/hicolor/32x32/apps RENAME net.kuribo64.melonDS.png) +install(FILES ../../../icon/melon_48x48.png DESTINATION ${CMAKE_INSTALL_PREFIX}/share/icons/hicolor/48x48/apps RENAME net.kuribo64.melonDS.png) +install(FILES ../../../icon/melon_64x64.png DESTINATION ${CMAKE_INSTALL_PREFIX}/share/icons/hicolor/64x64/apps RENAME net.kuribo64.melonDS.png) +install(FILES ../../../icon/melon_128x128.png DESTINATION ${CMAKE_INSTALL_PREFIX}/share/icons/hicolor/128x128/apps RENAME net.kuribo64.melonDS.png) +install(FILES ../../../icon/melon_256x256.png DESTINATION ${CMAKE_INSTALL_PREFIX}/share/icons/hicolor/256x256/apps RENAME net.kuribo64.melonDS.png) +install(FILES ../../../romlist.bin DESTINATION ${CMAKE_INSTALL_PREFIX}/share/melonDS) +install(TARGETS melonDS RUNTIME DESTINATION ${CMAKE_INSTALL_PREFIX}/bin) diff --git a/src/frontend/qt_sdl/EmuSettingsDialog.cpp b/src/frontend/qt_sdl/EmuSettingsDialog.cpp new file mode 100644 index 0000000..5c2efc0 --- /dev/null +++ b/src/frontend/qt_sdl/EmuSettingsDialog.cpp @@ -0,0 +1,147 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <stdio.h> +#include <QFileDialog> +#include <QMessageBox> + +#include "types.h" +#include "Platform.h" +#include "Config.h" +#include "PlatformConfig.h" + +#include "EmuSettingsDialog.h" +#include "ui_EmuSettingsDialog.h" + + +EmuSettingsDialog* EmuSettingsDialog::currentDlg = nullptr; + +extern char* EmuDirectory; + + +EmuSettingsDialog::EmuSettingsDialog(QWidget* parent) : QDialog(parent), ui(new Ui::EmuSettingsDialog) +{ +    ui->setupUi(this); +    setAttribute(Qt::WA_DeleteOnClose); + +    ui->txtBIOS9Path->setText(Config::BIOS9Path); +    ui->txtBIOS7Path->setText(Config::BIOS7Path); +    ui->txtFirmwarePath->setText(Config::FirmwarePath); +    ui->chkDirectBoot->setChecked(Config::DirectBoot != 0); +} + +EmuSettingsDialog::~EmuSettingsDialog() +{ +    delete ui; +} + +void EmuSettingsDialog::verifyFirmware() +{ +    // verify the firmware +    // +    // there are dumps of an old hacked firmware floating around on the internet +    // and those are problematic +    // the hack predates WFC, and, due to this, any game that alters the WFC +    // access point data will brick that firmware due to it having critical +    // data in the same area. it has the same problem on hardware. +    // +    // but this should help stop users from reporting that issue over and over +    // again, when the issue is not from melonDS but from their firmware dump. +    // +    // I don't know about all the firmware hacks in existence, but the one I +    // looked at has 0x180 bytes from the header repeated at 0x3FC80, but +    // bytes 0x0C-0x14 are different. + +    char filename[1024]; +    strncpy(filename, ui->txtFirmwarePath->text().toStdString().c_str(), 1023); filename[1023] = '\0'; +    FILE* f = Platform::OpenLocalFile(filename, "rb"); +    u8 chk1[0x180], chk2[0x180]; + +    fseek(f, 0, SEEK_SET); +    fread(chk1, 1, 0x180, f); +    fseek(f, -0x380, SEEK_END); +    fread(chk2, 1, 0x180, f); + +    memset(&chk1[0x0C], 0, 8); +    memset(&chk2[0x0C], 0, 8); + +    fclose(f); + +    if (!memcmp(chk1, chk2, 0x180)) +    { +        QMessageBox::warning((QWidget*)this->parent(), +                      "Problematic firmware dump", +                      "You are using an old hacked firmware dump.\n" +                      "Firmware boot will stop working if you run any game that alters WFC settings.\n\n" +                      "Note that the issue is not from melonDS, it would also happen on an actual DS."); +    } +} + +void EmuSettingsDialog::on_EmuSettingsDialog_accepted() +{ +    verifyFirmware(); + +    strncpy(Config::BIOS9Path, ui->txtBIOS9Path->text().toStdString().c_str(), 1023); Config::BIOS9Path[1023] = '\0'; +    strncpy(Config::BIOS7Path, ui->txtBIOS7Path->text().toStdString().c_str(), 1023); Config::BIOS7Path[1023] = '\0'; +    strncpy(Config::FirmwarePath, ui->txtFirmwarePath->text().toStdString().c_str(), 1023); Config::FirmwarePath[1023] = '\0'; +    Config::DirectBoot = ui->chkDirectBoot->isChecked() ? 1:0; +    Config::Save(); + +    closeDlg(); +} + +void EmuSettingsDialog::on_EmuSettingsDialog_rejected() +{ +    closeDlg(); +} + +void EmuSettingsDialog::on_btnBIOS9Browse_clicked() +{ +    QString file = QFileDialog::getOpenFileName(this, +                                                "Select DS-mode ARM9 BIOS...", +                                                EmuDirectory, +                                                "BIOS files (*.bin *.rom);;Any file (*.*)"); + +    if (file.isEmpty()) return; + +    ui->txtBIOS9Path->setText(file); +} + +void EmuSettingsDialog::on_btnBIOS7Browse_clicked() +{ +    QString file = QFileDialog::getOpenFileName(this, +                                                "Select DS-mode ARM7 BIOS...", +                                                EmuDirectory, +                                                "BIOS files (*.bin *.rom);;Any file (*.*)"); + +    if (file.isEmpty()) return; + +    ui->txtBIOS7Path->setText(file); +} + +void EmuSettingsDialog::on_btnFirmwareBrowse_clicked() +{ +    QString file = QFileDialog::getOpenFileName(this, +                                                "Select DS-mode firmware...", +                                                EmuDirectory, +                                                "Firmware files (*.bin *.rom);;Any file (*.*)"); + +    if (file.isEmpty()) return; + +    ui->txtFirmwarePath->setText(file); +} diff --git a/src/frontend/qt_sdl/EmuSettingsDialog.h b/src/frontend/qt_sdl/EmuSettingsDialog.h new file mode 100644 index 0000000..7378641 --- /dev/null +++ b/src/frontend/qt_sdl/EmuSettingsDialog.h @@ -0,0 +1,67 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef EMUSETTINGSDIALOG_H +#define EMUSETTINGSDIALOG_H + +#include <QDialog> + +namespace Ui { class EmuSettingsDialog; } +class EmuSettingsDialog; + +class EmuSettingsDialog : public QDialog +{ +    Q_OBJECT + +public: +    explicit EmuSettingsDialog(QWidget* parent); +    ~EmuSettingsDialog(); + +    static EmuSettingsDialog* currentDlg; +    static EmuSettingsDialog* openDlg(QWidget* parent) +    { +        if (currentDlg) +        { +            currentDlg->activateWindow(); +            return currentDlg; +        } + +        currentDlg = new EmuSettingsDialog(parent); +        currentDlg->show(); +        return currentDlg; +    } +    static void closeDlg() +    { +        currentDlg = nullptr; +    } + +private slots: +    void on_EmuSettingsDialog_accepted(); +    void on_EmuSettingsDialog_rejected(); + +    void on_btnBIOS9Browse_clicked(); +    void on_btnBIOS7Browse_clicked(); +    void on_btnFirmwareBrowse_clicked(); + +private: +    void verifyFirmware(); + +    Ui::EmuSettingsDialog* ui; +}; + +#endif // EMUSETTINGSDIALOG_H diff --git a/src/frontend/qt_sdl/EmuSettingsDialog.ui b/src/frontend/qt_sdl/EmuSettingsDialog.ui new file mode 100644 index 0000000..c70c3a2 --- /dev/null +++ b/src/frontend/qt_sdl/EmuSettingsDialog.ui @@ -0,0 +1,188 @@ +<?xml version="1.0" encoding="UTF-8"?> +<ui version="4.0"> + <class>EmuSettingsDialog</class> + <widget class="QDialog" name="EmuSettingsDialog"> +  <property name="geometry"> +   <rect> +    <x>0</x> +    <y>0</y> +    <width>490</width> +    <height>217</height> +   </rect> +  </property> +  <property name="sizePolicy"> +   <sizepolicy hsizetype="Fixed" vsizetype="Fixed"> +    <horstretch>0</horstretch> +    <verstretch>0</verstretch> +   </sizepolicy> +  </property> +  <property name="windowTitle"> +   <string>Emu settings - melonDS</string> +  </property> +  <layout class="QVBoxLayout" name="verticalLayout"> +   <property name="sizeConstraint"> +    <enum>QLayout::SetFixedSize</enum> +   </property> +   <item> +    <widget class="QGroupBox" name="groupBox"> +     <property name="title"> +      <string>DS mode</string> +     </property> +     <layout class="QGridLayout" name="gridLayout_2"> +      <item row="0" column="1"> +       <widget class="QLineEdit" name="txtBIOS9Path"> +        <property name="sizePolicy"> +         <sizepolicy hsizetype="Expanding" vsizetype="Fixed"> +          <horstretch>0</horstretch> +          <verstretch>0</verstretch> +         </sizepolicy> +        </property> +        <property name="minimumSize"> +         <size> +          <width>290</width> +          <height>0</height> +         </size> +        </property> +        <property name="statusTip"> +         <string/> +        </property> +        <property name="whatsThis"> +         <string><html><head/><body><p>DS-mode ARM9 BIOS</p><p>Size should be 4 KB</p></body></html></string> +        </property> +       </widget> +      </item> +      <item row="2" column="0"> +       <widget class="QLabel" name="label_3"> +        <property name="text"> +         <string>DS firmware:</string> +        </property> +       </widget> +      </item> +      <item row="1" column="0"> +       <widget class="QLabel" name="label_2"> +        <property name="text"> +         <string>DS ARM7 BIOS:</string> +        </property> +       </widget> +      </item> +      <item row="0" column="0"> +       <widget class="QLabel" name="label"> +        <property name="text"> +         <string>DS ARM9 BIOS:</string> +        </property> +       </widget> +      </item> +      <item row="0" column="2"> +       <widget class="QPushButton" name="btnBIOS9Browse"> +        <property name="sizePolicy"> +         <sizepolicy hsizetype="Minimum" vsizetype="Fixed"> +          <horstretch>0</horstretch> +          <verstretch>0</verstretch> +         </sizepolicy> +        </property> +        <property name="text"> +         <string>Browse...</string> +        </property> +        <property name="autoDefault"> +         <bool>true</bool> +        </property> +       </widget> +      </item> +      <item row="1" column="1"> +       <widget class="QLineEdit" name="txtBIOS7Path"> +        <property name="whatsThis"> +         <string><html><head/><body><p>DS-mode ARM7 BIOS</p><p>Size should be 16 KB</p></body></html></string> +        </property> +       </widget> +      </item> +      <item row="1" column="2"> +       <widget class="QPushButton" name="btnBIOS7Browse"> +        <property name="text"> +         <string>Browse...</string> +        </property> +       </widget> +      </item> +      <item row="2" column="1"> +       <widget class="QLineEdit" name="txtFirmwarePath"> +        <property name="whatsThis"> +         <string><html><head/><body><p>DS-mode firmware</p><p><br/></p><p>Possible firmwares:</p><p>* 128 KB: DS-mode firmware from a DSi or 3DS. Not bootable.</p><p>* 256 KB: regular DS firmware.</p><p>* 512 KB: iQue DS firmware.</p></body></html></string> +        </property> +       </widget> +      </item> +      <item row="2" column="2"> +       <widget class="QPushButton" name="btnFirmwareBrowse"> +        <property name="text"> +         <string>Browse...</string> +        </property> +       </widget> +      </item> +     </layout> +    </widget> +   </item> +   <item> +    <widget class="QGroupBox" name="groupBox_2"> +     <property name="title"> +      <string>Startup</string> +     </property> +     <layout class="QGridLayout" name="gridLayout"> +      <item row="0" column="0"> +       <widget class="QCheckBox" name="chkDirectBoot"> +        <property name="whatsThis"> +         <string><html><head/><body><p>When loading a ROM, completely skip the regular boot process (&quot;Nintendo DS&quot; screen) to boot the ROM directly.</p><p><br/></p><p>Note: if your firmware dump isn't bootable, the ROM will be booted directly regardless of this setting.</p></body></html></string> +        </property> +        <property name="text"> +         <string>Boot game directly</string> +        </property> +       </widget> +      </item> +     </layout> +    </widget> +   </item> +   <item> +    <widget class="QDialogButtonBox" name="buttonBox"> +     <property name="orientation"> +      <enum>Qt::Horizontal</enum> +     </property> +     <property name="standardButtons"> +      <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set> +     </property> +    </widget> +   </item> +  </layout> + </widget> + <resources/> + <connections> +  <connection> +   <sender>buttonBox</sender> +   <signal>accepted()</signal> +   <receiver>EmuSettingsDialog</receiver> +   <slot>accept()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>248</x> +     <y>254</y> +    </hint> +    <hint type="destinationlabel"> +     <x>157</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> +  <connection> +   <sender>buttonBox</sender> +   <signal>rejected()</signal> +   <receiver>EmuSettingsDialog</receiver> +   <slot>reject()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>316</x> +     <y>260</y> +    </hint> +    <hint type="destinationlabel"> +     <x>286</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> + </connections> +</ui> diff --git a/src/frontend/qt_sdl/Input.cpp b/src/frontend/qt_sdl/Input.cpp new file mode 100644 index 0000000..84d20ad --- /dev/null +++ b/src/frontend/qt_sdl/Input.cpp @@ -0,0 +1,244 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <QKeyEvent> +#include <SDL2/SDL.h> + +#include "Input.h" +#include "PlatformConfig.h" + + +namespace Input +{ + +int JoystickID; +SDL_Joystick* Joystick = nullptr; + +u32 KeyInputMask, JoyInputMask; +u32 KeyHotkeyMask, JoyHotkeyMask; +u32 HotkeyMask, LastHotkeyMask; +u32 HotkeyPress, HotkeyRelease; + +u32 InputMask; + + +void Init() +{ +    KeyInputMask = 0xFFF; +    JoyInputMask = 0xFFF; +    InputMask = 0xFFF; + +    KeyHotkeyMask = 0; +    JoyHotkeyMask = 0; +    HotkeyMask = 0; +    LastHotkeyMask = 0; +} + + +void OpenJoystick() +{ +    if (Joystick) SDL_JoystickClose(Joystick); + +    int num = SDL_NumJoysticks(); +    if (num < 1) +    { +        Joystick = nullptr; +        return; +    } + +    if (JoystickID >= num) +        JoystickID = 0; + +    Joystick = SDL_JoystickOpen(JoystickID); +} + +void CloseJoystick() +{ +    if (Joystick) +    { +        SDL_JoystickClose(Joystick); +        Joystick = nullptr; +    } +} + + +int GetEventKeyVal(QKeyEvent* event) +{ +    int key = event->key(); +    int mod = event->modifiers(); +    bool ismod = (key == Qt::Key_Control || +                  key == Qt::Key_Alt || +                  key == Qt::Key_AltGr || +                  key == Qt::Key_Shift || +                  key == Qt::Key_Meta); + +    if (!ismod) +        key |= mod; +    else if (Input::IsRightModKey(event)) +        key |= (1<<31); + +    return key; +} + +void KeyPress(QKeyEvent* event) +{ +    int keyHK = GetEventKeyVal(event); +    int keyKP = keyHK & ~event->modifiers(); + +    for (int i = 0; i < 12; i++) +        if (keyKP == Config::KeyMapping[i]) +            KeyInputMask &= ~(1<<i); + +    for (int i = 0; i < HK_MAX; i++) +        if (keyHK == Config::HKKeyMapping[i]) +            KeyHotkeyMask |= (1<<i); +} + +void KeyRelease(QKeyEvent* event) +{ +    int keyHK = GetEventKeyVal(event); +    int keyKP = keyHK & ~event->modifiers(); + +    for (int i = 0; i < 12; i++) +        if (keyKP == Config::KeyMapping[i]) +            KeyInputMask |= (1<<i); + +    for (int i = 0; i < HK_MAX; i++) +        if (keyHK == Config::HKKeyMapping[i]) +            KeyHotkeyMask &= ~(1<<i); +} + + +bool JoystickButtonDown(int val) +{ +    if (val == -1) return false; + +    bool hasbtn = ((val & 0xFFFF) != 0xFFFF); + +    if (hasbtn) +    { +        if (val & 0x100) +        { +            int hatnum = (val >> 4) & 0xF; +            int hatdir = val & 0xF; +            Uint8 hatval = SDL_JoystickGetHat(Joystick, hatnum); + +            bool pressed = false; +            if      (hatdir == 0x1) pressed = (hatval & SDL_HAT_UP); +            else if (hatdir == 0x4) pressed = (hatval & SDL_HAT_DOWN); +            else if (hatdir == 0x2) pressed = (hatval & SDL_HAT_RIGHT); +            else if (hatdir == 0x8) pressed = (hatval & SDL_HAT_LEFT); + +            if (pressed) return true; +        } +        else +        { +            int btnnum = val & 0xFFFF; +            Uint8 btnval = SDL_JoystickGetButton(Joystick, btnnum); + +            if (btnval) return true; +        } +    } + +    if (val & 0x10000) +    { +        int axisnum = (val >> 24) & 0xF; +        int axisdir = (val >> 20) & 0xF; +        Sint16 axisval = SDL_JoystickGetAxis(Joystick, axisnum); + +        switch (axisdir) +        { +        case 0: // positive +            if (axisval > 16384) return true; +            break; + +        case 1: // negative +            if (axisval < -16384) return true; +            break; + +        case 2: // trigger +            if (axisval > 0) return true; +            break; +        } +    } + +    return false; +} + +void Process() +{ +    SDL_JoystickUpdate(); + +    if (Joystick) +    { +        if (!SDL_JoystickGetAttached(Joystick)) +        { +            SDL_JoystickClose(Joystick); +            Joystick = NULL; +        } +    } +    if (!Joystick && (SDL_NumJoysticks() > 0)) +    { +        JoystickID = Config::JoystickID; +        OpenJoystick(); +    } + +    JoyInputMask = 0xFFF; +    for (int i = 0; i < 12; i++) +        if (JoystickButtonDown(Config::JoyMapping[i])) +            JoyInputMask &= ~(1<<i); + +    InputMask = KeyInputMask & JoyInputMask; + +    JoyHotkeyMask = 0; +    for (int i = 0; i < HK_MAX; i++) +        if (JoystickButtonDown(Config::HKJoyMapping[i])) +            JoyHotkeyMask |= (1<<i); + +    HotkeyMask = KeyHotkeyMask | JoyHotkeyMask; +    HotkeyPress = HotkeyMask & ~LastHotkeyMask; +    HotkeyRelease = LastHotkeyMask & ~HotkeyMask; +    LastHotkeyMask = HotkeyMask; +} + + +bool HotkeyDown(int id)     { return HotkeyMask    & (1<<id); } +bool HotkeyPressed(int id)  { return HotkeyPress   & (1<<id); } +bool HotkeyReleased(int id) { return HotkeyRelease & (1<<id); } + + +// TODO: MacOS version of this! +// distinguish between left and right modifier keys (Ctrl, Alt, Shift) +// Qt provides no real cross-platform way to do this, so here we go +// for Windows and Linux we can distinguish via scancodes (but both +// provide different scancodes) +#ifdef __WIN32__ +bool IsRightModKey(QKeyEvent* event) +{ +    quint32 scan = event->nativeScanCode(); +    return (scan == 0x11D || scan == 0x138 || scan == 0x36); +} +#else +bool IsRightModKey(QKeyEvent* event) +{ +    quint32 scan = event->nativeScanCode(); +    return (scan == 0x69 || scan == 0x6C || scan == 0x3E); +} +#endif + +} diff --git a/src/frontend/qt_sdl/Input.h b/src/frontend/qt_sdl/Input.h new file mode 100644 index 0000000..14e7ea8 --- /dev/null +++ b/src/frontend/qt_sdl/Input.h @@ -0,0 +1,51 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef INPUT_H +#define INPUT_H + +#include "types.h" + +namespace Input +{ + +extern int JoystickID; +extern SDL_Joystick* Joystick; + +extern u32 InputMask; + +void Init(); + +// set joystickID before calling openJoystick() +void OpenJoystick(); +void CloseJoystick(); + +void KeyPress(QKeyEvent* event); +void KeyRelease(QKeyEvent* event); + +void Process(); + +bool HotkeyDown(int id); +bool HotkeyPressed(int id); +bool HotkeyReleased(int id); + +bool IsRightModKey(QKeyEvent* event); + +} + +#endif // INPUT_H diff --git a/src/frontend/qt_sdl/InputConfigDialog.cpp b/src/frontend/qt_sdl/InputConfigDialog.cpp new file mode 100644 index 0000000..81baa65 --- /dev/null +++ b/src/frontend/qt_sdl/InputConfigDialog.cpp @@ -0,0 +1,494 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <QGroupBox> +#include <QLabel> +#include <QKeyEvent> + +#include <SDL2/SDL.h> + +#include "types.h" +#include "Config.h" +#include "PlatformConfig.h" + +#include "Input.h" +#include "InputConfigDialog.h" +#include "ui_InputConfigDialog.h" + + +InputConfigDialog* InputConfigDialog::currentDlg = nullptr; + +const int dskeyorder[12] = {0, 1, 10, 11, 5, 4, 6, 7, 9, 8, 2, 3}; +const char* dskeylabels[12] = {"A", "B", "X", "Y", "Left", "Right", "Up", "Down", "L", "R", "Select", "Start"}; + +const int hk_addons[] = +{ +    HK_SolarSensorIncrease, +    HK_SolarSensorDecrease, +}; + +const char* hk_addons_labels[] = +{ +    "[Boktai] Sunlight + ", +    "[Boktai] Sunlight - ", +}; + +const int hk_general[] = +{ +    HK_Pause, +    HK_Reset, +    HK_FastForward, +    HK_FastForwardToggle, +    HK_Lid, +    HK_Mic, +}; + +const char* hk_general_labels[] = +{ +    "Pause/resume", +    "Reset", +    "Fast forward", +    "Toggle FPS limit", +    "Close/open lid", +    "Microphone", +}; + + +InputConfigDialog::InputConfigDialog(QWidget* parent) : QDialog(parent), ui(new Ui::InputConfigDialog) +{ +    ui->setupUi(this); +    setAttribute(Qt::WA_DeleteOnClose); + +    for (int i = 0; i < 12; i++) +    { +        keypadKeyMap[i] = Config::KeyMapping[dskeyorder[i]]; +        keypadJoyMap[i] = Config::JoyMapping[dskeyorder[i]]; +    } + +    for (int i = 0; i < 2; i++) +    { +        addonsKeyMap[i] = Config::HKKeyMapping[hk_addons[i]]; +        addonsJoyMap[i] = Config::HKJoyMapping[hk_addons[i]]; +    } + +    for (int i = 0; i < 6; i++) +    { +        hkGeneralKeyMap[i] = Config::HKKeyMapping[hk_general[i]]; +        hkGeneralJoyMap[i] = Config::HKJoyMapping[hk_general[i]]; +    } + +    populatePage(ui->tabInput, 12, dskeylabels, keypadKeyMap, keypadJoyMap); +    populatePage(ui->tabAddons, 2, hk_addons_labels, addonsKeyMap, addonsJoyMap); +    populatePage(ui->tabHotkeysGeneral, 6, hk_general_labels, hkGeneralKeyMap, hkGeneralJoyMap); + +    int njoy = SDL_NumJoysticks(); +    if (njoy > 0) +    { +        for (int i = 0; i < njoy; i++) +        { +            const char* name = SDL_JoystickNameForIndex(i); +            ui->cbxJoystick->addItem(QString(name)); +        } +        ui->cbxJoystick->setCurrentIndex(Input::JoystickID); +    } +    else +    { +        ui->cbxJoystick->addItem("(no joysticks available)"); +        ui->cbxJoystick->setEnabled(false); +    } +} + +InputConfigDialog::~InputConfigDialog() +{ +    delete ui; +} + +void InputConfigDialog::populatePage(QWidget* page, int num, const char** labels, int* keymap, int* joymap) +{ +    // kind of a hack +    bool ishotkey = (page != ui->tabInput); + +    QHBoxLayout* main_layout = new QHBoxLayout(); + +    QGroupBox* group; +    QGridLayout* group_layout; + +    group = new QGroupBox("Keyboard mappings:"); +    main_layout->addWidget(group); +    group_layout = new QGridLayout(); +    group_layout->setSpacing(1); +    for (int i = 0; i < num; i++) +    { +        QLabel* label = new QLabel(QString(labels[i])+":"); +        KeyMapButton* btn = new KeyMapButton(&keymap[i], ishotkey); + +        group_layout->addWidget(label, i, 0); +        group_layout->addWidget(btn, i, 1); +    } +    group_layout->setRowStretch(num, 1); +    group->setLayout(group_layout); +    group->setMinimumWidth(275); + +    group = new QGroupBox("Joystick mappings:"); +    main_layout->addWidget(group); +    group_layout = new QGridLayout(); +    group_layout->setSpacing(1); +    for (int i = 0; i < num; i++) +    { +        QLabel* label = new QLabel(QString(labels[i])+":"); +        JoyMapButton* btn = new JoyMapButton(&joymap[i], ishotkey); + +        group_layout->addWidget(label, i, 0); +        group_layout->addWidget(btn, i, 1); +    } +    group_layout->setRowStretch(num, 1); +    group->setLayout(group_layout); +    group->setMinimumWidth(275); + +    page->setLayout(main_layout); +} + +void InputConfigDialog::on_InputConfigDialog_accepted() +{ +    for (int i = 0; i < 12; i++) +    { +        Config::KeyMapping[dskeyorder[i]] = keypadKeyMap[i]; +        Config::JoyMapping[dskeyorder[i]] = keypadJoyMap[i]; +    } + +    for (int i = 0; i < 2; i++) +    { +        Config::HKKeyMapping[hk_addons[i]] = addonsKeyMap[i]; +        Config::HKJoyMapping[hk_addons[i]] = addonsJoyMap[i]; +    } + +    for (int i = 0; i < 6; i++) +    { +        Config::HKKeyMapping[hk_general[i]] = hkGeneralKeyMap[i]; +        Config::HKJoyMapping[hk_general[i]] = hkGeneralJoyMap[i]; +    } + +    Config::JoystickID = Input::JoystickID; +    Config::Save(); + +    closeDlg(); +} + +void InputConfigDialog::on_InputConfigDialog_rejected() +{ +    Input::JoystickID = Config::JoystickID; +    Input::OpenJoystick(); + +    closeDlg(); +} + +void InputConfigDialog::on_cbxJoystick_currentIndexChanged(int id) +{ +    // prevent a spurious change +    if (ui->cbxJoystick->count() < 2) return; + +    Input::JoystickID = id; +    Input::OpenJoystick(); +} + + +KeyMapButton::KeyMapButton(int* mapping, bool hotkey) : QPushButton() +{ +    this->mapping = mapping; +    this->isHotkey = hotkey; + +    setCheckable(true); +    setText(mappingText()); + +    connect(this, &KeyMapButton::clicked, this, &KeyMapButton::onClick); +} + +KeyMapButton::~KeyMapButton() +{ +} + +void KeyMapButton::keyPressEvent(QKeyEvent* event) +{ +    if (!isChecked()) return QPushButton::keyPressEvent(event); + +    printf("KEY PRESSED = %08X %08X | %08X %08X %08X\n", event->key(), event->modifiers(), event->nativeVirtualKey(), event->nativeModifiers(), event->nativeScanCode()); + +    int key = event->key(); +    int mod = event->modifiers(); +    bool ismod = (key == Qt::Key_Control || +                  key == Qt::Key_Alt || +                  key == Qt::Key_AltGr || +                  key == Qt::Key_Shift || +                  key == Qt::Key_Meta); + +    if (!mod) +    { +        if (key == Qt::Key_Escape) { click(); return; } +        if (key == Qt::Key_Backspace) { *mapping = -1; click(); return; } +    } + +    if (isHotkey) +    { +        if (ismod) +            return; +    } + +    if (!ismod) +        key |= mod; +    else if (Input::IsRightModKey(event)) +        key |= (1<<31); + +    *mapping = key; +    click(); +} + +void KeyMapButton::focusOutEvent(QFocusEvent* event) +{ +    if (isChecked()) +    { +        // if we lost the focus while mapping, consider it 'done' +        click(); +    } + +    QPushButton::focusOutEvent(event); +} + +void KeyMapButton::onClick() +{ +    if (isChecked()) +    { +        setText("[press key]"); +    } +    else +    { +        setText(mappingText()); +    } +} + +QString KeyMapButton::mappingText() +{ +    int key = *mapping; + +    if (key == -1) return "None"; + +    QString isright = (key & (1<<31)) ? "Right " : "Left "; +    key &= ~(1<<31); + +    switch (key) +    { +    case Qt::Key_Control: return isright + "Ctrl"; +    case Qt::Key_Alt:     return "Alt"; +    case Qt::Key_AltGr:   return "AltGr"; +    case Qt::Key_Shift:   return isright + "Shift"; +    case Qt::Key_Meta:    return "Meta"; +    } + +    QKeySequence seq(key); +    QString ret = seq.toString(); + +    // weak attempt at detecting garbage key names +    if (ret.length() == 2 && ret[0].unicode() > 0xFF) +        return QString("[%1]").arg(key, 8, 16); + +    return ret.replace("&", "&&"); +} + + +JoyMapButton::JoyMapButton(int* mapping, bool hotkey) : QPushButton() +{ +    this->mapping = mapping; +    this->isHotkey = hotkey; + +    setCheckable(true); +    setText(mappingText()); + +    connect(this, &JoyMapButton::clicked, this, &JoyMapButton::onClick); + +    timerID = 0; +} + +JoyMapButton::~JoyMapButton() +{ +} + +void JoyMapButton::keyPressEvent(QKeyEvent* event) +{ +    if (!isChecked()) return QPushButton::keyPressEvent(event); + +    int key = event->key(); +    int mod = event->modifiers(); + +    if (!mod) +    { +        if (key == Qt::Key_Escape) { click(); return; } +        if (key == Qt::Key_Backspace) { *mapping = -1; click(); return; } +    } +} + +void JoyMapButton::focusOutEvent(QFocusEvent* event) +{ +    if (isChecked()) +    { +        // if we lost the focus while mapping, consider it 'done' +        click(); +    } + +    QPushButton::focusOutEvent(event); +} + +void JoyMapButton::timerEvent(QTimerEvent* event) +{ +    SDL_Joystick* joy = Input::Joystick; +    if (!joy) { click(); return; } +    if (!SDL_JoystickGetAttached(joy)) { click(); return; } + +    int oldmap; +    if (*mapping == -1) oldmap = 0xFFFF; +    else                oldmap = *mapping; + +    int nbuttons = SDL_JoystickNumButtons(joy); +    for (int i = 0; i < nbuttons; i++) +    { +        if (SDL_JoystickGetButton(joy, i)) +        { +            *mapping = (oldmap & 0xFFFF0000) | i; +            click(); +            return; +        } +    } + +    int nhats = SDL_JoystickNumHats(joy); +    if (nhats > 16) nhats = 16; +    for (int i = 0; i < nhats; i++) +    { +        Uint8 blackhat = SDL_JoystickGetHat(joy, i); +        if (blackhat) +        { +            if      (blackhat & 0x1) blackhat = 0x1; +            else if (blackhat & 0x2) blackhat = 0x2; +            else if (blackhat & 0x4) blackhat = 0x4; +            else                     blackhat = 0x8; + +            *mapping = (oldmap & 0xFFFF0000) | 0x100 | blackhat | (i << 4); +            click(); +            return; +        } +    } + +    int naxes = SDL_JoystickNumAxes(joy); +    if (naxes > 16) naxes = 16; +    for (int i = 0; i < naxes; i++) +    { +        Sint16 axisval = SDL_JoystickGetAxis(joy, i); +        int diff = abs(axisval - axesRest[i]); + +        if (axesRest[i] < -16384 && axisval >= 0) +        { +            *mapping = (oldmap & 0xFFFF) | 0x10000 | (2 << 20) | (i << 24); +            click(); +            return; +        } +        else if (diff > 16384) +        { +            int axistype; +            if (axisval > 0) axistype = 0; +            else             axistype = 1; + +            *mapping = (oldmap & 0xFFFF) | 0x10000 | (axistype << 20) | (i << 24); +            click(); +            return; +        } +    } +} + +void JoyMapButton::onClick() +{ +    if (isChecked()) +    { +        setText("[press button/axis]"); +        timerID = startTimer(50); + +        memset(axesRest, 0, sizeof(axesRest)); +        if (Input::Joystick && SDL_JoystickGetAttached(Input::Joystick)) +        { +            int naxes = SDL_JoystickNumAxes(Input::Joystick); +            if (naxes > 16) naxes = 16; +            for (int a = 0; a < naxes; a++) +            { +                axesRest[a] = SDL_JoystickGetAxis(Input::Joystick, a); +            } +        } +    } +    else +    { +        setText(mappingText()); +        if (timerID) { killTimer(timerID); timerID = 0; } +    } +} + +QString JoyMapButton::mappingText() +{ +    int id = *mapping; + +    if (id == -1) return "None"; + +    bool hasbtn = ((id & 0xFFFF) != 0xFFFF); +    QString str; + +    if (hasbtn) +    { +        if (id & 0x100) +        { +            int hatnum = ((id >> 4) & 0xF) + 1; + +            switch (id & 0xF) +            { +            case 0x1: str = "Hat %1 up"; break; +            case 0x2: str = "Hat %1 right"; break; +            case 0x4: str = "Hat %1 down"; break; +            case 0x8: str = "Hat %1 left"; break; +            } + +            str = str.arg(hatnum); +        } +        else +        { +            str = QString("Button %1").arg((id & 0xFFFF) + 1); +        } +    } +    else +    { +        str = ""; +    } + +    if (id & 0x10000) +    { +        int axisnum = ((id >> 24) & 0xF) + 1; + +        if (hasbtn) str += " / "; + +        switch ((id >> 20) & 0xF) +        { +        case 0: str += QString("Axis %1 +").arg(axisnum); break; +        case 1: str += QString("Axis %1 -").arg(axisnum); break; +        case 2: str += QString("Trigger %1").arg(axisnum); break; +        } +    } + +    return str; +} diff --git a/src/frontend/qt_sdl/InputConfigDialog.h b/src/frontend/qt_sdl/InputConfigDialog.h new file mode 100644 index 0000000..de57414 --- /dev/null +++ b/src/frontend/qt_sdl/InputConfigDialog.h @@ -0,0 +1,123 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef INPUTCONFIGDIALOG_H +#define INPUTCONFIGDIALOG_H + +#include <QDialog> +#include <QPushButton> + +namespace Ui { class InputConfigDialog; } +class InputConfigDialog; + +class InputConfigDialog : public QDialog +{ +    Q_OBJECT + +public: +    explicit InputConfigDialog(QWidget* parent); +    ~InputConfigDialog(); + +    static InputConfigDialog* currentDlg; +    static InputConfigDialog* openDlg(QWidget* parent) +    { +        if (currentDlg) +        { +            currentDlg->activateWindow(); +            return currentDlg; +        } + +        currentDlg = new InputConfigDialog(parent); +        currentDlg->open(); +        return currentDlg; +    } +    static void closeDlg() +    { +        currentDlg = nullptr; +    } + +private slots: +    void on_InputConfigDialog_accepted(); +    void on_InputConfigDialog_rejected(); + +    void on_cbxJoystick_currentIndexChanged(int id); + +private: +    void populatePage(QWidget* page, int num, const char** labels, int* keymap, int* joymap); + +    Ui::InputConfigDialog* ui; + +    int keypadKeyMap[12],   keypadJoyMap[12]; +    int addonsKeyMap[2],    addonsJoyMap[2]; +    int hkGeneralKeyMap[6], hkGeneralJoyMap[6]; +}; + + +class KeyMapButton : public QPushButton +{ +    Q_OBJECT + +public: +    explicit KeyMapButton(int* mapping, bool hotkey); +    ~KeyMapButton(); + +protected: +    void keyPressEvent(QKeyEvent* event) override; +    void focusOutEvent(QFocusEvent* event) override; + +    bool focusNextPrevChild(bool next) override { return false; } + +private slots: +    void onClick(); + +private: +    QString mappingText(); + +    int* mapping; +    bool isHotkey; +}; + +class JoyMapButton : public QPushButton +{ +    Q_OBJECT + +public: +    explicit JoyMapButton(int* mapping, bool hotkey); +    ~JoyMapButton(); + +protected: +    void keyPressEvent(QKeyEvent* event) override; +    void focusOutEvent(QFocusEvent* event) override; +    void timerEvent(QTimerEvent* event) override; + +    bool focusNextPrevChild(bool next) override { return false; } + +private slots: +    void onClick(); + +private: +    QString mappingText(); + +    int* mapping; +    bool isHotkey; + +    int timerID; +    int axesRest[16]; +}; + +#endif // INPUTCONFIGDIALOG_H diff --git a/src/frontend/qt_sdl/InputConfigDialog.ui b/src/frontend/qt_sdl/InputConfigDialog.ui new file mode 100644 index 0000000..655da16 --- /dev/null +++ b/src/frontend/qt_sdl/InputConfigDialog.ui @@ -0,0 +1,131 @@ +<?xml version="1.0" encoding="UTF-8"?> +<ui version="4.0"> + <class>InputConfigDialog</class> + <widget class="QDialog" name="InputConfigDialog"> +  <property name="geometry"> +   <rect> +    <x>0</x> +    <y>0</y> +    <width>488</width> +    <height>365</height> +   </rect> +  </property> +  <property name="windowTitle"> +   <string>Input and hotkeys - melonDS</string> +  </property> +  <layout class="QVBoxLayout" name="verticalLayout"> +   <property name="sizeConstraint"> +    <enum>QLayout::SetFixedSize</enum> +   </property> +   <item> +    <widget class="QTabWidget" name="tabWidget"> +     <property name="currentIndex"> +      <number>0</number> +     </property> +     <widget class="QWidget" name="tabInput"> +      <attribute name="title"> +       <string>DS keypad</string> +      </attribute> +     </widget> +     <widget class="QWidget" name="tabAddons"> +      <attribute name="title"> +       <string>Add-ons</string> +      </attribute> +     </widget> +     <widget class="QWidget" name="tabHotkeysGeneral"> +      <attribute name="title"> +       <string>General hotkeys</string> +      </attribute> +     </widget> +    </widget> +   </item> +   <item> +    <layout class="QHBoxLayout" name="horizontalLayout"> +     <property name="leftMargin"> +      <number>0</number> +     </property> +     <property name="topMargin"> +      <number>0</number> +     </property> +     <property name="rightMargin"> +      <number>0</number> +     </property> +     <property name="bottomMargin"> +      <number>0</number> +     </property> +     <item> +      <widget class="QLabel" name="label"> +       <property name="sizePolicy"> +        <sizepolicy hsizetype="Preferred" vsizetype="Preferred"> +         <horstretch>0</horstretch> +         <verstretch>0</verstretch> +        </sizepolicy> +       </property> +       <property name="text"> +        <string>Joystick:</string> +       </property> +      </widget> +     </item> +     <item> +      <widget class="QComboBox" name="cbxJoystick"> +       <property name="sizePolicy"> +        <sizepolicy hsizetype="MinimumExpanding" vsizetype="Fixed"> +         <horstretch>0</horstretch> +         <verstretch>0</verstretch> +        </sizepolicy> +       </property> +       <property name="whatsThis"> +        <string><html><head/><body><p>Selects which joystick will be used for joystick input, if any is present.</p></body></html></string> +       </property> +      </widget> +     </item> +    </layout> +   </item> +   <item> +    <widget class="QDialogButtonBox" name="buttonBox"> +     <property name="orientation"> +      <enum>Qt::Horizontal</enum> +     </property> +     <property name="standardButtons"> +      <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set> +     </property> +    </widget> +   </item> +  </layout> + </widget> + <resources/> + <connections> +  <connection> +   <sender>buttonBox</sender> +   <signal>accepted()</signal> +   <receiver>InputConfigDialog</receiver> +   <slot>accept()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>248</x> +     <y>254</y> +    </hint> +    <hint type="destinationlabel"> +     <x>157</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> +  <connection> +   <sender>buttonBox</sender> +   <signal>rejected()</signal> +   <receiver>InputConfigDialog</receiver> +   <slot>reject()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>316</x> +     <y>260</y> +    </hint> +    <hint type="destinationlabel"> +     <x>286</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> + </connections> +</ui> diff --git a/src/frontend/qt_sdl/LAN_PCap.cpp b/src/frontend/qt_sdl/LAN_PCap.cpp new file mode 100644 index 0000000..ce278bc --- /dev/null +++ b/src/frontend/qt_sdl/LAN_PCap.cpp @@ -0,0 +1,387 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +// direct LAN interface. Currently powered by libpcap, may change. + +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <SDL2/SDL.h> +#include <pcap/pcap.h> +#include "../Wifi.h" +#include "LAN_PCap.h" +#include "PlatformConfig.h" + +#ifdef __WIN32__ +	#include <iphlpapi.h> +#else +	#include <sys/types.h> +	#include <ifaddrs.h> +	#include <netinet/in.h> +	#include <linux/if_packet.h> +#endif + + +// welp +#ifndef PCAP_OPENFLAG_PROMISCUOUS +#define PCAP_OPENFLAG_PROMISCUOUS 1 +#endif + + +#define DECL_PCAP_FUNC(ret, name, args, args2) \ +    typedef ret (*type_##name) args; \ +    type_##name ptr_##name = NULL; \ +    ret name args { return ptr_##name args2; } + +DECL_PCAP_FUNC(int, pcap_findalldevs, (pcap_if_t** alldevs, char* errbuf), (alldevs,errbuf)) +DECL_PCAP_FUNC(void, pcap_freealldevs, (pcap_if_t* alldevs), (alldevs)) +DECL_PCAP_FUNC(pcap_t*, pcap_open_live, (const char* src, int snaplen, int flags, int readtimeout, char* errbuf), (src,snaplen,flags,readtimeout,errbuf)) +DECL_PCAP_FUNC(void, pcap_close, (pcap_t* dev), (dev)) +DECL_PCAP_FUNC(int, pcap_setnonblock, (pcap_t* dev, int nonblock, char* errbuf), (dev,nonblock,errbuf)) +DECL_PCAP_FUNC(int, pcap_sendpacket, (pcap_t* dev, const u_char* data, int len), (dev,data,len)) +DECL_PCAP_FUNC(int, pcap_dispatch, (pcap_t* dev, int num, pcap_handler callback, u_char* data), (dev,num,callback,data)) +DECL_PCAP_FUNC(const u_char*, pcap_next, (pcap_t* dev, struct pcap_pkthdr* hdr), (dev,hdr)) + + +namespace LAN_PCap +{ + +const char* PCapLibNames[] = +{ +#ifdef __WIN32__ +    // TODO: name for npcap in non-WinPCap mode +    "wpcap.dll", +#else +    // Linux lib names +    "libpcap.so.1", +    "libpcap.so", +#endif +    NULL +}; + +AdapterData* Adapters = NULL; +int NumAdapters = 0; + +void* PCapLib = NULL; +pcap_t* PCapAdapter = NULL; +AdapterData* PCapAdapterData; + +u8 PacketBuffer[2048]; +int PacketLen; +volatile int RXNum; + + +#define LOAD_PCAP_FUNC(sym) \ +    ptr_##sym = (type_##sym)SDL_LoadFunction(lib, #sym); \ +    if (!ptr_##sym) return false; + +bool TryLoadPCap(void* lib) +{ +    LOAD_PCAP_FUNC(pcap_findalldevs) +    LOAD_PCAP_FUNC(pcap_freealldevs) +    LOAD_PCAP_FUNC(pcap_open_live) +    LOAD_PCAP_FUNC(pcap_close) +    LOAD_PCAP_FUNC(pcap_setnonblock) +    LOAD_PCAP_FUNC(pcap_sendpacket) +    LOAD_PCAP_FUNC(pcap_dispatch) +    LOAD_PCAP_FUNC(pcap_next) + +    return true; +} + +bool Init(bool open_adapter) +{ +    // TODO: how to deal with cases where an adapter is unplugged or changes config?? +    if (!PCapLib) +    { +        PCapLib = NULL; + +        for (int i = 0; PCapLibNames[i]; i++) +        { +            void* lib = SDL_LoadObject(PCapLibNames[i]); +            if (!lib) continue; + +            if (!TryLoadPCap(lib)) +            { +                SDL_UnloadObject(lib); +                continue; +            } + +            printf("PCap: lib %s, init successful\n", PCapLibNames[i]); +            PCapLib = lib; +            break; +        } + +        if (PCapLib == NULL) +        { +            printf("PCap: init failed\n"); +            return false; +        } +    } + +    PCapAdapter = NULL; +    PacketLen = 0; +    RXNum = 0; + +    NumAdapters = 0; + +    char errbuf[PCAP_ERRBUF_SIZE]; +    int ret; + +    pcap_if_t* alldevs; +    ret = pcap_findalldevs(&alldevs, errbuf); +    if (ret < 0 || alldevs == NULL) +    { +        printf("PCap: no devices available\n"); +        return false; +    } + +    pcap_if_t* dev = alldevs; +    while (dev) { NumAdapters++; dev = dev->next; } + +    Adapters = new AdapterData[NumAdapters]; +    memset(Adapters, 0, sizeof(AdapterData)*NumAdapters); + +    AdapterData* adata = &Adapters[0]; +    dev = alldevs; +    while (dev) +    { +        adata->Internal = dev; + +#ifdef __WIN32__ +        // hax +        int len = strlen(dev->name); +        len -= 12; if (len > 127) len = 127; +        strncpy(adata->DeviceName, &dev->name[12], len); +        adata->DeviceName[len] = '\0'; +#else +        strncpy(adata->DeviceName, dev->name, 127); +        adata->DeviceName[127] = '\0'; + +        strncpy(adata->FriendlyName, adata->DeviceName, 127); +        adata->FriendlyName[127] = '\0'; +#endif // __WIN32__ + +        dev = dev->next; +        adata++; +    } + +#ifdef __WIN32__ + +    ULONG bufsize = 16384; +    IP_ADAPTER_ADDRESSES* buf = (IP_ADAPTER_ADDRESSES*)HeapAlloc(GetProcessHeap(), 0, bufsize); +    ULONG uret = GetAdaptersAddresses(AF_INET, 0, NULL, buf, &bufsize); +    if (uret == ERROR_BUFFER_OVERFLOW) +    { +        HeapFree(GetProcessHeap(), 0, buf); +        buf = (IP_ADAPTER_ADDRESSES*)HeapAlloc(GetProcessHeap(), 0, bufsize); +        uret = GetAdaptersAddresses(AF_INET, 0, NULL, buf, &bufsize); +    } +    if (uret != ERROR_SUCCESS) +    { +        printf("GetAdaptersAddresses() shat itself: %08X\n", uret); +        return false; +    } + +    for (int i = 0; i < NumAdapters; i++) +    { +        adata = &Adapters[i]; +        IP_ADAPTER_ADDRESSES* addr = buf; +        while (addr) +        { +            if (strcmp(addr->AdapterName, adata->DeviceName)) +            { +                addr = addr->Next; +                continue; +            } + +            WideCharToMultiByte(CP_UTF8, 0, addr->FriendlyName, 127, adata->FriendlyName, 127, NULL, NULL); +            adata->FriendlyName[127] = '\0'; + +            WideCharToMultiByte(CP_UTF8, 0, addr->Description, 127, adata->Description, 127, NULL, NULL); +            adata->Description[127] = '\0'; + +            if (addr->PhysicalAddressLength != 6) +            { +                printf("weird MAC addr length %d for %s\n", addr->PhysicalAddressLength, addr->AdapterName); +            } +            else +                memcpy(adata->MAC, addr->PhysicalAddress, 6); + +            IP_ADAPTER_UNICAST_ADDRESS* ipaddr = addr->FirstUnicastAddress; +            while (ipaddr) +            { +                SOCKADDR* sa = ipaddr->Address.lpSockaddr; +                if (sa->sa_family == AF_INET) +                { +                    struct in_addr sa4 = ((sockaddr_in*)sa)->sin_addr; +                    memcpy(adata->IP_v4, &sa4, 4); +                } + +                ipaddr = ipaddr->Next; +            } + +            break; +        } +    } + +    HeapFree(GetProcessHeap(), 0, buf); + +#else + +    struct ifaddrs* addrs; +    if (getifaddrs(&addrs) != 0) +    { +        printf("getifaddrs() shat itself :(\n"); +        return false; +    } + +    for (int i = 0; i < NumAdapters; i++) +    { +        adata = &Adapters[i]; +        struct ifaddrs* curaddr = addrs; +        while (curaddr) +        { +            if (strcmp(curaddr->ifa_name, adata->DeviceName)) +            { +                curaddr = curaddr->ifa_next; +                continue; +            } + +            if (!curaddr->ifa_addr) +            { +                printf("Device (%s) does not have an address :/\n", curaddr->ifa_name); +                curaddr = curaddr->ifa_next; +                continue; +            } + +            u16 af = curaddr->ifa_addr->sa_family; +            if (af == AF_INET) +            { +                struct sockaddr_in* sa = (sockaddr_in*)curaddr->ifa_addr; +                memcpy(adata->IP_v4, &sa->sin_addr, 4); +            } +            else if (af == AF_PACKET) +            { +                struct sockaddr_ll* sa = (sockaddr_ll*)curaddr->ifa_addr; +                if (sa->sll_halen != 6) +                    printf("weird MAC length %d for %s\n", sa->sll_halen, curaddr->ifa_name); +                else +                    memcpy(adata->MAC, sa->sll_addr, 6); +            } + +            curaddr = curaddr->ifa_next; +        } +    } + +    freeifaddrs(addrs); + +#endif // __WIN32__ + +    if (!open_adapter) return true; +    if (PCapAdapter) pcap_close(PCapAdapter); + +    // open pcap device +    PCapAdapterData = &Adapters[0]; +    for (int i = 0; i < NumAdapters; i++) +    { +        if (!strncmp(Adapters[i].DeviceName, Config::LANDevice, 128)) +            PCapAdapterData = &Adapters[i]; +    } + +    dev = (pcap_if_t*)PCapAdapterData->Internal; +    PCapAdapter = pcap_open_live(dev->name, 2048, PCAP_OPENFLAG_PROMISCUOUS, 1, errbuf); +    if (!PCapAdapter) +    { +        printf("PCap: failed to open adapter %s\n", errbuf); +        return false; +    } + +    pcap_freealldevs(alldevs); + +    if (pcap_setnonblock(PCapAdapter, 1, errbuf) < 0) +    { +        printf("PCap: failed to set nonblocking mode\n"); +        pcap_close(PCapAdapter); PCapAdapter = NULL; +        return false; +    } + +    return true; +} + +void DeInit() +{ +    if (PCapLib) +    { +        if (PCapAdapter) +        { +            pcap_close(PCapAdapter); +            PCapAdapter = NULL; +        } + +        SDL_UnloadObject(PCapLib); +        PCapLib = NULL; +    } +} + + +void RXCallback(u_char* blarg, const struct pcap_pkthdr* header, const u_char* data) +{ +    while (RXNum > 0); + +    if (header->len > 2048-64) return; + +    PacketLen = header->len; +    memcpy(PacketBuffer, data, PacketLen); +    RXNum = 1; +} + +int SendPacket(u8* data, int len) +{ +    if (PCapAdapter == NULL) +        return 0; + +    if (len > 2048) +    { +        printf("LAN_SendPacket: error: packet too long (%d)\n", len); +        return 0; +    } + +    pcap_sendpacket(PCapAdapter, data, len); +    // TODO: check success +    return len; +} + +int RecvPacket(u8* data) +{ +    if (PCapAdapter == NULL) +        return 0; + +    int ret = 0; +    if (RXNum > 0) +    { +        memcpy(data, PacketBuffer, PacketLen); +        ret = PacketLen; +        RXNum = 0; +    } + +    pcap_dispatch(PCapAdapter, 1, RXCallback, NULL); +    return ret; +} + +} diff --git a/src/frontend/qt_sdl/LAN_PCap.h b/src/frontend/qt_sdl/LAN_PCap.h new file mode 100644 index 0000000..250b8e9 --- /dev/null +++ b/src/frontend/qt_sdl/LAN_PCap.h @@ -0,0 +1,53 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef LAN_PCAP_H +#define LAN_PCAP_H + +#include "../types.h" + +namespace LAN_PCap +{ + +typedef struct +{ +    char DeviceName[128]; +    char FriendlyName[128]; +    char Description[128]; + +    u8 MAC[6]; +    u8 IP_v4[4]; + +    void* Internal; + +} AdapterData; + + +extern AdapterData* Adapters; +extern int NumAdapters; + + +bool Init(bool open_adapter); +void DeInit(); + +int SendPacket(u8* data, int len); +int RecvPacket(u8* data); + +} + +#endif // LAN_PCAP_H diff --git a/src/frontend/qt_sdl/LAN_Socket.cpp b/src/frontend/qt_sdl/LAN_Socket.cpp new file mode 100644 index 0000000..c6fbd4b --- /dev/null +++ b/src/frontend/qt_sdl/LAN_Socket.cpp @@ -0,0 +1,1145 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +// indirect LAN interface, powered by BSD sockets. + +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include "../Wifi.h" +#include "LAN_Socket.h" +#include "../Config.h" + +#ifdef __WIN32__ +	#include <winsock2.h> +	#include <ws2tcpip.h> +	#define socket_t    SOCKET +	#define sockaddr_t  SOCKADDR +#else +	#include <unistd.h> +	#include <arpa/inet.h> +	#include <netinet/in.h> +	#include <sys/types.h> +	#include <sys/select.h> +	#include <sys/socket.h> +	#include <netdb.h> +	#define socket_t    int +	#define sockaddr_t  struct sockaddr +	#define closesocket close +#endif + +#ifndef INVALID_SOCKET +#define INVALID_SOCKET  (socket_t)-1 +#endif + + +namespace LAN_Socket +{ + +const u32 kSubnet   = 0x0A400000; +const u32 kServerIP = kSubnet | 0x01; +const u32 kDNSIP    = kSubnet | 0x02; +const u32 kClientIP = kSubnet | 0x10; + +const u8 kServerMAC[6] = {0x00, 0xAB, 0x33, 0x28, 0x99, 0x44}; +const u8 kDNSMAC[6]    = {0x00, 0xAB, 0x33, 0x28, 0x99, 0x55}; + +u8 PacketBuffer[2048]; +int PacketLen; +volatile int RXNum; + +u16 IPv4ID; + + +// TODO: UDP sockets +// * use FIFO list +// * assign new socket when seeing new IP/port + + +typedef struct +{ +    u8 DestIP[4]; +    u16 SourcePort; +    u16 DestPort; + +    u32 SeqNum; // sequence number for incoming frames +    u32 AckNum; + +    // 0: unused +    // 1: connected +    u8 Status; + +    socket_t Backend; + +} TCPSocket; + +typedef struct +{ +    u8 DestIP[4]; +    u16 SourcePort; +    u16 DestPort; + +    socket_t Backend; +    struct sockaddr_in BackendAddr; + +} UDPSocket; + +TCPSocket TCPSocketList[16]; +UDPSocket UDPSocketList[4]; + +int UDPSocketID = 0; + + +bool Init() +{ +    // TODO: how to deal with cases where an adapter is unplugged or changes config?? +    //if (PCapLib) return true; + +    //Lib = NULL; +    PacketLen = 0; +    RXNum = 0; + +    IPv4ID = 1; + +    memset(TCPSocketList, 0, sizeof(TCPSocketList)); +    memset(UDPSocketList, 0, sizeof(UDPSocketList)); + +    UDPSocketID = 0; + +    return true; +} + +void DeInit() +{ +    for (int i = 0; i < (sizeof(TCPSocketList)/sizeof(TCPSocket)); i++) +    { +        TCPSocket* sock = &TCPSocketList[i]; +        if (sock->Backend) closesocket(sock->Backend); +    } + +    for (int i = 0; i < (sizeof(UDPSocketList)/sizeof(UDPSocket)); i++) +    { +        UDPSocket* sock = &UDPSocketList[i]; +        if (sock->Backend) closesocket(sock->Backend); +    } +} + + +void FinishUDPFrame(u8* data, int len) +{ +    u8* ipheader = &data[0xE]; +    u8* udpheader = &data[0x22]; + +    // lengths +    *(u16*)&ipheader[2] = htons(len - 0xE); +    *(u16*)&udpheader[4] = htons(len - (0xE + 0x14)); + +    // IP checksum +    u32 tmp = 0; + +    for (int i = 0; i < 20; i += 2) +        tmp += ntohs(*(u16*)&ipheader[i]); +    while (tmp >> 16) +        tmp = (tmp & 0xFFFF) + (tmp >> 16); +    tmp ^= 0xFFFF; +    *(u16*)&ipheader[10] = htons(tmp); + +    // UDP checksum +    // (note: normally not mandatory, but some older sgIP versions require it) +    tmp = 0; +    tmp += ntohs(*(u16*)&ipheader[12]); +    tmp += ntohs(*(u16*)&ipheader[14]); +    tmp += ntohs(*(u16*)&ipheader[16]); +    tmp += ntohs(*(u16*)&ipheader[18]); +    tmp += ntohs(0x1100); +    tmp += (len-0x22); +    for (u8* i = udpheader; i < &udpheader[len-0x23]; i += 2) +        tmp += ntohs(*(u16*)i); +    if (len & 1) +        tmp += ntohs((u_short)udpheader[len-0x23]); +    while (tmp >> 16) +        tmp = (tmp & 0xFFFF) + (tmp >> 16); +    tmp ^= 0xFFFF; +    if (tmp == 0) tmp = 0xFFFF; +    *(u16*)&udpheader[6] = htons(tmp); +} + +void FinishTCPFrame(u8* data, int len) +{ +    u8* ipheader = &data[0xE]; +    u8* tcpheader = &data[0x22]; + +    // lengths +    *(u16*)&ipheader[2] = htons(len - 0xE); + +    // IP checksum +    u32 tmp = 0; + +    for (int i = 0; i < 20; i += 2) +        tmp += ntohs(*(u16*)&ipheader[i]); +    while (tmp >> 16) +        tmp = (tmp & 0xFFFF) + (tmp >> 16); +    tmp ^= 0xFFFF; +    *(u16*)&ipheader[10] = htons(tmp); + +    u32 tcplen = ntohs(*(u16*)&ipheader[2]) - 0x14; + +    // TCP checksum +    tmp = 0; +    tmp += ntohs(*(u16*)&ipheader[12]); +    tmp += ntohs(*(u16*)&ipheader[14]); +    tmp += ntohs(*(u16*)&ipheader[16]); +    tmp += ntohs(*(u16*)&ipheader[18]); +    tmp += ntohs(0x0600); +    tmp += tcplen; +    for (u8* i = tcpheader; i < &tcpheader[tcplen-1]; i += 2) +        tmp += ntohs(*(u16*)i); +    if (tcplen & 1) +        tmp += ntohs((u_short)tcpheader[tcplen-1]); +    while (tmp >> 16) +        tmp = (tmp & 0xFFFF) + (tmp >> 16); +    tmp ^= 0xFFFF; +    *(u16*)&tcpheader[16] = htons(tmp); +} + + +void HandleDHCPFrame(u8* data, int len) +{ +    u8 type = 0xFF; + +    u32 transid = *(u32*)&data[0x2E]; + +    u8* options = &data[0x11A]; +    for (;;) +    { +        if (options >= &data[len]) break; +        u8 opt = *options++; +        if (opt == 255) break; + +        u8 len = *options++; +        switch (opt) +        { +        case 53: // frame type +            type = options[0]; +            break; +        } + +        options += len; +    } + +    if (type == 0xFF) +    { +        printf("DHCP: bad frame\n"); +        return; +    } + +    printf("DHCP: frame type %d, transid %08X\n", type, transid); + +    if (type == 1 || // discover +        type == 3)   // request +    { +        u8 resp[512]; +        u8* out = &resp[0]; + +        // ethernet +        memcpy(out, &data[6], 6); out += 6; +        memcpy(out, kServerMAC, 6); out += 6; +        *(u16*)out = htons(0x0800); out += 2; + +        // IP +        u8* ipheader = out; +        *out++ = 0x45; +        *out++ = 0x00; +        *(u16*)out = 0; out += 2; // total length +        *(u16*)out = htons(IPv4ID); out += 2; IPv4ID++; +        *out++ = 0x00; +        *out++ = 0x00; +        *out++ = 0x80; // TTL +        *out++ = 0x11; // protocol (UDP) +        *(u16*)out = 0; out += 2; // checksum +        *(u32*)out = htonl(kServerIP); out += 4; // source IP +        if (type == 1) +        { +            *(u32*)out = htonl(0xFFFFFFFF); out += 4; // destination IP +        } +        else if (type == 3) +        { +            *(u32*)out = htonl(kClientIP); out += 4; // destination IP +        } + +        // UDP +        u8* udpheader = out; +        *(u16*)out = htons(67); out += 2; // source port +        *(u16*)out = htons(68); out += 2; // destination port +        *(u16*)out = 0; out += 2; // length +        *(u16*)out = 0; out += 2; // checksum + +        // DHCP +        u8* body = out; +        *out++ = 0x02; +        *out++ = 0x01; +        *out++ = 0x06; +        *out++ = 0x00; +        *(u32*)out = transid; out += 4; +        *(u16*)out = 0; out += 2; // seconds elapsed +        *(u16*)out = 0; out += 2; +        *(u32*)out = htonl(0x00000000); out += 4; // client IP +        *(u32*)out = htonl(kClientIP); out += 4; // your IP +        *(u32*)out = htonl(kServerIP); out += 4; // server IP +        *(u32*)out = htonl(0x00000000); out += 4; // gateway IP +        memcpy(out, &data[6], 6); out += 6; +        memset(out, 0, 10); out += 10; +        memset(out, 0, 192); out += 192; +        *(u32*)out = 0x63538263; out += 4; // DHCP magic + +        // DHCP options +        *out++ = 53; *out++ = 1; +        *out++ = (type==1) ? 2 : 5; // DHCP type: offer/ack +        *out++ = 1; *out++ = 4; +        *(u32*)out = htonl(0xFFFFFF00); out += 4; // subnet mask +        *out++ = 3; *out++ = 4; +        *(u32*)out = htonl(kServerIP); out += 4; // router +        *out++ = 51; *out++ = 4; +        *(u32*)out = htonl(442030); out += 4; // lease time +        *out++ = 54; *out++ = 4; +        *(u32*)out = htonl(kServerIP); out += 4; // DHCP server +        *out++ = 6; *out++ = 4; +        *(u32*)out = htonl(kDNSIP); out += 4; // DNS (hax) + +        *out++ = 0xFF; +        memset(out, 0, 20); out += 20; + +        u32 framelen = (u32)(out - &resp[0]); +        if (framelen & 1) { *out++ = 0; framelen++; } +        FinishUDPFrame(resp, framelen); + +        // TODO: if there is already a packet queued, this will overwrite it +        // that being said, this will only happen during DHCP setup, so probably +        // not a big deal + +        PacketLen = framelen; +        memcpy(PacketBuffer, resp, PacketLen); +        RXNum = 1; +    } +} + +void HandleDNSFrame(u8* data, int len) +{ +    u8* ipheader = &data[0xE]; +    u8* udpheader = &data[0x22]; +    u8* dnsbody = &data[0x2A]; + +    u32 srcip = ntohl(*(u32*)&ipheader[12]); +    u16 srcport = ntohs(*(u16*)&udpheader[0]); + +    u16 id = ntohs(*(u16*)&dnsbody[0]); +    u16 flags = ntohs(*(u16*)&dnsbody[2]); +    u16 numquestions = ntohs(*(u16*)&dnsbody[4]); +    u16 numanswers = ntohs(*(u16*)&dnsbody[6]); +    u16 numauth = ntohs(*(u16*)&dnsbody[8]); +    u16 numadd = ntohs(*(u16*)&dnsbody[10]); + +    printf("DNS: ID=%04X, flags=%04X, Q=%d, A=%d, auth=%d, add=%d\n", +           id, flags, numquestions, numanswers, numauth, numadd); + +    // for now we only take 'simple' DNS requests +    if (flags & 0x8000) return; +    if (numquestions != 1 || numanswers != 0) return; + +    u8 resp[1024]; +    u8* out = &resp[0]; + +    // ethernet +    memcpy(out, &data[6], 6); out += 6; +    memcpy(out, kServerMAC, 6); out += 6; +    *(u16*)out = htons(0x0800); out += 2; + +    // IP +    u8* resp_ipheader = out; +    *out++ = 0x45; +    *out++ = 0x00; +    *(u16*)out = 0; out += 2; // total length +    *(u16*)out = htons(IPv4ID); out += 2; IPv4ID++; +    *out++ = 0x00; +    *out++ = 0x00; +    *out++ = 0x80; // TTL +    *out++ = 0x11; // protocol (UDP) +    *(u16*)out = 0; out += 2; // checksum +    *(u32*)out = htonl(kDNSIP); out += 4; // source IP +    *(u32*)out = htonl(srcip); out += 4; // destination IP + +    // UDP +    u8* resp_udpheader = out; +    *(u16*)out = htons(53); out += 2; // source port +    *(u16*)out = htons(srcport); out += 2; // destination port +    *(u16*)out = 0; out += 2; // length +    *(u16*)out = 0; out += 2; // checksum + +    // DNS +    u8* resp_body = out; +    *(u16*)out = htons(id); out += 2; // ID +    *(u16*)out = htons(0x8000); out += 2; // flags +    *(u16*)out = htons(numquestions); out += 2; // num questions +    *(u16*)out = htons(numquestions); out += 2; // num answers +    *(u16*)out = 0; out += 2; // num authority +    *(u16*)out = 0; out += 2; // num additional + +    u32 curoffset = 12; +    for (u16 i = 0; i < numquestions; i++) +    { +        if (curoffset >= (len-0x2A)) return; + +        u8 bitlength = 0; +        while ((bitlength = dnsbody[curoffset++]) != 0) +            curoffset += bitlength; + +        curoffset += 4; +    } + +    u32 qlen = curoffset-12; +    if (qlen > 512) return; +    memcpy(out, &dnsbody[12], qlen); out += qlen; + +    curoffset = 12; +	for (u16 i = 0; i < numquestions; i++) +	{ +		// assemble the requested domain name +		u8 bitlength = 0; +		char domainname[256] = ""; int o = 0; +		while ((bitlength = dnsbody[curoffset++]) != 0) +		{ +		    if ((o+bitlength) >= 255) +            { +                // welp. atleast try not to explode. +                domainname[o++] = '\0'; +                break; +            } + +			strncpy(&domainname[o], (const char *)&dnsbody[curoffset], bitlength); +			o += bitlength; + +			curoffset += bitlength; +			if (dnsbody[curoffset] != 0) +				domainname[o++] = '.'; +            else +                domainname[o++] = '\0'; +		} + +		u16 type = ntohs(*(u16*)&dnsbody[curoffset]); +		u16 cls = ntohs(*(u16*)&dnsbody[curoffset+2]); + +		printf("- q%d: %04X %04X %s", i, type, cls, domainname); + +		// get answer +		struct addrinfo dns_hint; +		struct addrinfo* dns_res; +		u32 addr_res; + +		memset(&dns_hint, 0, sizeof(dns_hint)); +		dns_hint.ai_family = AF_INET; // TODO: other address types (INET6, etc) +		if (getaddrinfo(domainname, "0", &dns_hint, &dns_res) == 0) +        { +            struct addrinfo* p = dns_res; +            while (p) +            { +                struct sockaddr_in* addr = (struct sockaddr_in*)p->ai_addr; +                /*printf(" -> %d.%d.%d.%d", +                       addr->sin_addr.S_un.S_un_b.s_b1, addr->sin_addr.S_un.S_un_b.s_b2, +                       addr->sin_addr.S_un.S_un_b.s_b3, addr->sin_addr.S_un.S_un_b.s_b4);*/ + +                //addr_res = addr->sin_addr.S_un.S_addr; +                addr_res = *(u32*)&addr->sin_addr; +                p = p->ai_next; +            } +        } +        else +        { +            printf(" shat itself :("); +            addr_res = 0; +        } + +		printf("\n"); +		curoffset += 4; + +		// TODO: betterer support +		// (under which conditions does the C00C marker work?) +		*(u16*)out = htons(0xC00C); out += 2; +		*(u16*)out = htons(type); out += 2; +		*(u16*)out = htons(cls); out += 2; +		*(u32*)out = htonl(3600); out += 4; // TTL (hardcoded for now) +		*(u16*)out = htons(4); out += 2; // address length +		*(u32*)out = addr_res; out += 4; // address +    } + +    u32 framelen = (u32)(out - &resp[0]); +    if (framelen & 1) { *out++ = 0; framelen++; } +    FinishUDPFrame(resp, framelen); + +    // TODO: if there is already a packet queued, this will overwrite it +    // that being said, this will only happen during DHCP setup, so probably +    // not a big deal + +    PacketLen = framelen; +    memcpy(PacketBuffer, resp, PacketLen); +    RXNum = 1; +} + +void UDP_BuildIncomingFrame(UDPSocket* sock, u8* data, int len) +{ +    u8 resp[2048]; +    u8* out = &resp[0]; + +    if (len > 1536) return; + +    // ethernet +    memcpy(out, Wifi::GetMAC(), 6); out += 6; // hurf +    memcpy(out, kServerMAC, 6); out += 6; +    *(u16*)out = htons(0x0800); out += 2; + +    // IP +    u8* resp_ipheader = out; +    *out++ = 0x45; +    *out++ = 0x00; +    *(u16*)out = 0; out += 2; // total length +    *(u16*)out = htons(IPv4ID); out += 2; IPv4ID++; +    *out++ = 0x00; +    *out++ = 0x00; +    *out++ = 0x80; // TTL +    *out++ = 0x11; // protocol (UDP) +    *(u16*)out = 0; out += 2; // checksum +    memcpy(out, sock->DestIP, 4); out += 4; // source IP +    *(u32*)out = htonl(kClientIP); out += 4; // destination IP + +    // UDP +    u8* resp_tcpheader = out; +    *(u16*)out = htons(sock->DestPort); out += 2; // source port +    *(u16*)out = htons(sock->SourcePort); out += 2; // destination port +    *(u16*)out = htons(len+8); out += 2; // length of header+data +    *(u16*)out = 0; out += 2; // checksum + +    memcpy(out, data, len); out += len; + +    u32 framelen = (u32)(out - &resp[0]); +    FinishUDPFrame(resp, framelen); + +    // TODO: if there is already a packet queued, this will overwrite it +    // that being said, this will only happen during DHCP setup, so probably +    // not a big deal + +    PacketLen = framelen; +    memcpy(PacketBuffer, resp, PacketLen); +    RXNum = 1; +} + +void HandleUDPFrame(u8* data, int len) +{ +    u8* ipheader = &data[0xE]; +    u8* udpheader = &data[0x22]; + +    // debug +    /*for (int j = 0; j < len; j += 16) +    { +        int rem = len - j; +        if (rem > 16) rem = 16; +        for (int i = 0; i < rem; i++) +        { +            printf("%02X ", data[i+j]); +        } +        printf("\n"); +    }*/ + +    u16 srcport = ntohs(*(u16*)&udpheader[0]); +    u16 dstport = ntohs(*(u16*)&udpheader[2]); + +    int sockid = -1; +    UDPSocket* sock; +    for (int i = 0; i < (sizeof(UDPSocketList)/sizeof(UDPSocket)); i++) +    { +        sock = &UDPSocketList[i]; +        if (sock->Backend != 0 && !memcmp(&sock->DestIP, &ipheader[16], 4) && +            sock->SourcePort == srcport && sock->DestPort == dstport) +        { +            sockid = i; +            break; +        } +    } + +    if (sockid == -1) +    { +        sockid = UDPSocketID; +        sock = &UDPSocketList[sockid]; + +        UDPSocketID++; +        if (UDPSocketID >= (sizeof(UDPSocketList)/sizeof(UDPSocket))) +            UDPSocketID = 0; + +        if (sock->Backend != 0) +        { +            printf("LANMAGIC: closing previous UDP socket #%d\n", sockid); +            closesocket(sock->Backend); +        } + +        sock->Backend = socket(AF_INET, SOCK_DGRAM, 0); + +        memcpy(sock->DestIP, &ipheader[16], 4); +        sock->SourcePort = srcport; +        sock->DestPort = dstport; + +        memset(&sock->BackendAddr, 0, sizeof(sock->BackendAddr)); +        sock->BackendAddr.sin_family = AF_INET; +        sock->BackendAddr.sin_port = htons(dstport); +        memcpy(&sock->BackendAddr.sin_addr, &ipheader[16], 4); +        /*if (bind(sock->Backend, (struct sockaddr*)&sock->BackendAddr, sizeof(sock->BackendAddr)) == -1) +        { +            printf("bind() shat itself :(\n"); +        }*/ + +        printf("LANMAGIC: opening UDP socket #%d to %d.%d.%d.%d:%d, srcport %d\n", +               sockid, +               ipheader[16], ipheader[17], ipheader[18], ipheader[19], +               dstport, srcport); +    } + +    u16 udplen = ntohs(*(u16*)&udpheader[4]) - 8; + +    printf("UDP: socket %d sending %d bytes\n", sockid, udplen); +    sendto(sock->Backend, (char*)&udpheader[8], udplen, 0, +           (struct sockaddr*)&sock->BackendAddr, sizeof(sock->BackendAddr)); +} + +void TCP_SYNACK(TCPSocket* sock, u8* data, int len) +{ +    u8 resp[128]; +    u8* out = &resp[0]; + +    u8* ipheader = &data[0xE]; +    u8* tcpheader = &data[0x22]; + +    u32 seqnum = htonl(*(u32*)&tcpheader[4]); +    seqnum++; +    sock->AckNum = seqnum; + +    //printf("SYNACK  SEQ=%08X|%08X\n", sock->SeqNum, sock->AckNum); + +    // ethernet +    memcpy(out, &data[6], 6); out += 6; +    memcpy(out, kServerMAC, 6); out += 6; +    *(u16*)out = htons(0x0800); out += 2; + +    // IP +    u8* resp_ipheader = out; +    *out++ = 0x45; +    *out++ = 0x00; +    *(u16*)out = 0; out += 2; // total length +    *(u16*)out = htons(IPv4ID); out += 2; IPv4ID++; +    *out++ = 0x00; +    *out++ = 0x00; +    *out++ = 0x80; // TTL +    *out++ = 0x06; // protocol (TCP) +    *(u16*)out = 0; out += 2; // checksum +    *(u32*)out = *(u32*)&ipheader[16]; out += 4; // source IP +    *(u32*)out = *(u32*)&ipheader[12]; out += 4; // destination IP + +    // TCP +    u8* resp_tcpheader = out; +    *(u16*)out = *(u16*)&tcpheader[2]; out += 2; // source port +    *(u16*)out = *(u16*)&tcpheader[0]; out += 2; // destination port +    *(u32*)out = htonl(sock->SeqNum); out += 4; sock->SeqNum++; // seq number +    *(u32*)out = htonl(seqnum); out += 4; // ack seq number +    *(u16*)out = htons(0x8012); out += 2; // flags (SYN+ACK) +    *(u16*)out = htons(0x7000); out += 2; // window size (uuuh) +    *(u16*)out = 0; out += 2; // checksum +    *(u16*)out = 0; out += 2; // urgent pointer + +    // TCP options +    *out++ = 0x02; *out++ = 0x04; // max segment size +    *(u16*)out = htons(0x05B4); out += 2; +    *out++ = 0x01; +    *out++ = 0x01; +    *out++ = 0x04; *out++ = 0x02; // SACK permitted +    *out++ = 0x01; +    *out++ = 0x03; *out++ = 0x03; // window size +    *out++ = 0x08; + +    u32 framelen = (u32)(out - &resp[0]); +    //if (framelen & 1) { *out++ = 0; framelen++; } +    FinishTCPFrame(resp, framelen); + +    // TODO: if there is already a packet queued, this will overwrite it +    // that being said, this will only happen during DHCP setup, so probably +    // not a big deal + +    PacketLen = framelen; +    memcpy(PacketBuffer, resp, PacketLen); +    RXNum = 1; +} + +void TCP_ACK(TCPSocket* sock, bool fin) +{ +    u8 resp[64]; +    u8* out = &resp[0]; + +    u16 flags       = 0x5010; +    if (fin) flags |= 0x0001; + +    //printf("%sACK  SEQ=%08X|%08X\n", fin?"FIN":"   ", sock->SeqNum, sock->AckNum); + +    // ethernet +    memcpy(out, Wifi::GetMAC(), 6); out += 6; +    memcpy(out, kServerMAC, 6); out += 6; +    *(u16*)out = htons(0x0800); out += 2; + +    // IP +    u8* resp_ipheader = out; +    *out++ = 0x45; +    *out++ = 0x00; +    *(u16*)out = 0; out += 2; // total length +    *(u16*)out = htons(IPv4ID); out += 2; IPv4ID++; +    *out++ = 0x00; +    *out++ = 0x00; +    *out++ = 0x80; // TTL +    *out++ = 0x06; // protocol (TCP) +    *(u16*)out = 0; out += 2; // checksum +    *(u32*)out = *(u32*)&sock->DestIP; out += 4; // source IP +    *(u32*)out = htonl(kClientIP); out += 4; // destination IP + +    // TCP +    u8* resp_tcpheader = out; +    *(u16*)out = htonl(sock->DestPort); out += 2; // source port +    *(u16*)out = htonl(sock->SourcePort); out += 2; // destination port +    *(u32*)out = htonl(sock->SeqNum); out += 4; // seq number +    *(u32*)out = htonl(sock->AckNum); out += 4; // ack seq number +    *(u16*)out = htons(flags); out += 2; // flags +    *(u16*)out = htons(0x7000); out += 2; // window size (uuuh) +    *(u16*)out = 0; out += 2; // checksum +    *(u16*)out = 0; out += 2; // urgent pointer + +    u32 framelen = (u32)(out - &resp[0]); +    //if (framelen & 1) { *out++ = 0; framelen++; } +    FinishTCPFrame(resp, framelen); + +    // TODO: if there is already a packet queued, this will overwrite it +    // that being said, this will only happen during DHCP setup, so probably +    // not a big deal + +    PacketLen = framelen; +    memcpy(PacketBuffer, resp, PacketLen); +    RXNum = 1; +} + +void TCP_BuildIncomingFrame(TCPSocket* sock, u8* data, int len) +{ +    u8 resp[2048]; +    u8* out = &resp[0]; + +    if (len > 1536) return; +//printf("INCOMING SEQ=%08X|%08X\n", sock->SeqNum, sock->AckNum); +    // ethernet +    memcpy(out, Wifi::GetMAC(), 6); out += 6; // hurf +    memcpy(out, kServerMAC, 6); out += 6; +    *(u16*)out = htons(0x0800); out += 2; + +    // IP +    u8* resp_ipheader = out; +    *out++ = 0x45; +    *out++ = 0x00; +    *(u16*)out = 0; out += 2; // total length +    *(u16*)out = htons(IPv4ID); out += 2; IPv4ID++; +    *out++ = 0x00; +    *out++ = 0x00; +    *out++ = 0x80; // TTL +    *out++ = 0x06; // protocol (TCP) +    *(u16*)out = 0; out += 2; // checksum +    memcpy(out, sock->DestIP, 4); out += 4; // source IP +    *(u32*)out = htonl(kClientIP); out += 4; // destination IP + +    // TCP +    u8* resp_tcpheader = out; +    *(u16*)out = htons(sock->DestPort); out += 2; // source port +    *(u16*)out = htons(sock->SourcePort); out += 2; // destination port +    *(u32*)out = htonl(sock->SeqNum); out += 4; // seq number +    *(u32*)out = htonl(sock->AckNum); out += 4; // ack seq number +    *(u16*)out = htons(0x5018); out += 2; // flags (ACK, PSH) +    *(u16*)out = htons(0x7000); out += 2; // window size (uuuh) +    *(u16*)out = 0; out += 2; // checksum +    *(u16*)out = 0; out += 2; // urgent pointer + +    memcpy(out, data, len); out += len; + +    u32 framelen = (u32)(out - &resp[0]); +    FinishTCPFrame(resp, framelen); + +    // TODO: if there is already a packet queued, this will overwrite it +    // that being said, this will only happen during DHCP setup, so probably +    // not a big deal + +    PacketLen = framelen; +    memcpy(PacketBuffer, resp, PacketLen); +    RXNum = 1; + +    sock->SeqNum += len; +} + +void HandleTCPFrame(u8* data, int len) +{ +    u8* ipheader = &data[0xE]; +    u8* tcpheader = &data[0x22]; + +    u16 srcport = ntohs(*(u16*)&tcpheader[0]); +    u16 dstport = ntohs(*(u16*)&tcpheader[2]); +    u16 flags = ntohs(*(u16*)&tcpheader[12]); + +    u32 tcpheaderlen = 4 * (flags >> 12); +    u32 tcplen = ntohs(*(u16*)&ipheader[2]) - 0x14; +    u32 tcpdatalen = tcplen - tcpheaderlen; + +    /*printf("tcpflags=%04X header=%d data=%d seq=%08X|%08X\n", +           flags, tcpheaderlen, tcpdatalen, +           ntohl(*(u32*)&tcpheader[4]), +           ntohl(*(u32*)&tcpheader[8]));*/ + +    if (flags & 0x002) // SYN +    { +        int sockid = -1; +        TCPSocket* sock; +        for (int i = 0; i < (sizeof(TCPSocketList)/sizeof(TCPSocket)); i++) +        { +            sock = &TCPSocketList[i]; +            if (sock->Status != 0 && !memcmp(&sock->DestIP, &ipheader[16], 4) && +                sock->SourcePort == srcport && sock->DestPort == dstport) +            { +                printf("LANMAGIC: duplicate TCP socket\n"); +                sockid = i; +                break; +            } +        } + +        if (sockid == -1) +        { +            for (int i = 0; i < (sizeof(TCPSocketList)/sizeof(TCPSocket)); i++) +            { +                sock = &TCPSocketList[i]; +                if (sock->Status == 0) +                { +                    sockid = i; +                    break; +                } +            } +        } + +        if (sockid == -1) +        { +            printf("LANMAGIC: !! TCP SOCKET LIST FULL\n"); +            return; +        } + +        printf("LANMAGIC: opening TCP socket #%d to %d.%d.%d.%d:%d, srcport %d\n", +               sockid, +               ipheader[16], ipheader[17], ipheader[18], ipheader[19], +               dstport, srcport); + +        // keep track of it +        sock->Status = 1; +        memcpy(sock->DestIP, &ipheader[16], 4); +        sock->DestPort = dstport; +        sock->SourcePort = srcport; +        sock->SeqNum = 0x13370000; +        sock->AckNum = 0; + +        // open backend socket +        if (!sock->Backend) +        { +            sock->Backend = socket(AF_INET, SOCK_STREAM, 0); +        } + +        struct sockaddr_in conn_addr; +        memset(&conn_addr, 0, sizeof(conn_addr)); +        conn_addr.sin_family = AF_INET; +        memcpy(&conn_addr.sin_addr, &ipheader[16], 4); +        conn_addr.sin_port = htons(dstport); +        if (connect(sock->Backend, (sockaddr*)&conn_addr, sizeof(conn_addr)) == -1) +        { +            printf("connect() shat itself :(\n"); +        } +        else +        { +            // acknowledge it +            TCP_SYNACK(sock, data, len); +        } +    } +    else +    { +        int sockid = -1; +        TCPSocket* sock; +        for (int i = 0; i < (sizeof(TCPSocketList)/sizeof(TCPSocket)); i++) +        { +            sock = &TCPSocketList[i]; +            if (sock->Status != 0 && !memcmp(&sock->DestIP, &ipheader[16], 4) && +                sock->SourcePort == srcport && sock->DestPort == dstport) +            { +                sockid = i; +                break; +            } +        } + +        if (sockid == -1) +        { +            printf("LANMAGIC: bad TCP packet\n"); +            return; +        } + +        // TODO: check those +        u32 seqnum = ntohl(*(u32*)&tcpheader[4]); +        u32 acknum = ntohl(*(u32*)&tcpheader[8]); +        sock->SeqNum = acknum; +        sock->AckNum = seqnum + tcpdatalen; + +        // send data over the socket +        if (tcpdatalen > 0) +        { +            u8* tcpdata = &tcpheader[tcpheaderlen]; + +            printf("TCP: socket %d sending %d bytes (flags=%04X)\n", sockid, tcpdatalen, flags); +            send(sock->Backend, (char*)tcpdata, tcpdatalen, 0); + +            // kind of a hack, there +            TCP_ACK(sock, false); +        } + +        if (flags & 0x001) // FIN +        { +            // TODO: timeout etc +            printf("TCP: socket %d closing\n", sockid); + +            sock->Status = 0; +            closesocket(sock->Backend); +            sock->Backend = 0; +        } +    } +} + +void HandleARPFrame(u8* data, int len) +{ +    u16 protocol = ntohs(*(u16*)&data[0x10]); +    if (protocol != 0x0800) return; + +    u16 op = ntohs(*(u16*)&data[0x14]); +    u32 targetip = ntohl(*(u32*)&data[0x26]); + +    // TODO: handle ARP to the client +    // this only handles ARP to the DHCP/router + +    if (op == 1) +    { +        // opcode 1=req 2=reply +        // sender MAC +        // sender IP +        // target MAC +        // target IP + +        const u8* targetmac; +        if (targetip == kServerIP)   targetmac = kServerMAC; +        else if (targetip == kDNSIP) targetmac = kDNSMAC; +        else return; + +        u8 resp[64]; +        u8* out = &resp[0]; + +        // ethernet +        memcpy(out, &data[6], 6); out += 6; +        memcpy(out, kServerMAC, 6); out += 6; +        *(u16*)out = htons(0x0806); out += 2; + +        // ARP +        *(u16*)out = htons(0x0001); out += 2; // hardware type +        *(u16*)out = htons(0x0800); out += 2; // protocol +        *out++ = 6; // MAC address size +        *out++ = 4; // IP address size +        *(u16*)out = htons(0x0002); out += 2; // opcode +        memcpy(out, targetmac, 6); out += 6; +        *(u32*)out = htonl(targetip); out += 4; +        memcpy(out, &data[0x16], 6+4); out += 6+4; + +        u32 framelen = (u32)(out - &resp[0]); + +        // TODO: if there is already a packet queued, this will overwrite it +        // that being said, this will only happen during DHCP setup, so probably +        // not a big deal + +        PacketLen = framelen; +        memcpy(PacketBuffer, resp, PacketLen); +        RXNum = 1; +    } +    else +    { +        printf("wat??\n"); +    } +} + +void HandlePacket(u8* data, int len) +{ +    u16 ethertype = ntohs(*(u16*)&data[0xC]); + +    if (ethertype == 0x0800) // IPv4 +    { +        u8 protocol = data[0x17]; +        if (protocol == 0x11) // UDP +        { +            u16 srcport = ntohs(*(u16*)&data[0x22]); +            u16 dstport = ntohs(*(u16*)&data[0x24]); +            if (srcport == 68 && dstport == 67) // DHCP +            { +                printf("LANMAGIC: DHCP packet\n"); +                return HandleDHCPFrame(data, len); +            } +            else if (dstport == 53 && htonl(*(u32*)&data[0x1E]) == kDNSIP) // DNS +            { +                printf("LANMAGIC: DNS packet\n"); +                return HandleDNSFrame(data, len); +            } + +            printf("LANMAGIC: UDP packet %d->%d\n", srcport, dstport); +            return HandleUDPFrame(data, len); +        } +        else if (protocol == 0x06) // TCP +        { +            printf("LANMAGIC: TCP packet\n"); +            return HandleTCPFrame(data, len); +        } +        else +            printf("LANMAGIC: unsupported IP protocol %02X\n", protocol); +    } +    else if (ethertype == 0x0806) // ARP +    { +        printf("LANMAGIC: ARP packet\n"); +        return HandleARPFrame(data, len); +    } +    else +        printf("LANMAGIC: unsupported ethernet type %04X\n", ethertype); +} + +int SendPacket(u8* data, int len) +{ +    if (len > 2048) +    { +        printf("LAN_SendPacket: error: packet too long (%d)\n", len); +        return 0; +    } + +    HandlePacket(data, len); +    return len; +} + +int RecvPacket(u8* data) +{ +    int ret = 0; +    if (RXNum > 0) +    { +        memcpy(data, PacketBuffer, PacketLen); +        ret = PacketLen; +        RXNum = 0; +    } + +    for (int i = 0; i < (sizeof(TCPSocketList)/sizeof(TCPSocket)); i++) +    { +        TCPSocket* sock = &TCPSocketList[i]; +        if (sock->Status != 1) continue; + +        fd_set fd; +        struct timeval tv; + +        FD_ZERO(&fd); +        FD_SET(sock->Backend, &fd); +        tv.tv_sec = 0; +        tv.tv_usec = 0; + +        if (!select(sock->Backend+1, &fd, 0, 0, &tv)) +        { +            continue; +        } + +        u8 recvbuf[1024]; +        int recvlen = recv(sock->Backend, (char*)recvbuf, 1024, 0); +        if (recvlen < 1) +        { +            if (recvlen == 0) +            { +                // socket has closed from the other side +                printf("TCP: socket %d closed from other side\n", i); +                sock->Status = 2; +                TCP_ACK(sock, true); +            } +            continue; +        } + +        printf("TCP: socket %d receiving %d bytes\n", i, recvlen); +        TCP_BuildIncomingFrame(sock, recvbuf, recvlen); + +        // debug +        /*for (int j = 0; j < recvlen; j += 16) +        { +            int rem = recvlen - j; +            if (rem > 16) rem = 16; +            for (int k = 0; k < rem; k++) +            { +                printf("%02X ", recvbuf[k+j]); +            } +            printf("\n"); +        }*/ + +        //recvlen = recv(sock->Backend, (char*)recvbuf, 1024, 0); +        //if (recvlen == 0) printf("it closed immediately after\n"); +    } + +    for (int i = 0; i < (sizeof(UDPSocketList)/sizeof(UDPSocket)); i++) +    { +        UDPSocket* sock = &UDPSocketList[i]; +        if (sock->Backend == 0) continue; + +        fd_set fd; +        struct timeval tv; + +        FD_ZERO(&fd); +        FD_SET(sock->Backend, &fd); +        tv.tv_sec = 0; +        tv.tv_usec = 0; + +        if (!select(sock->Backend+1, &fd, 0, 0, &tv)) +        { +            continue; +        } + +        u8 recvbuf[1024]; +        sockaddr_t fromAddr; +        socklen_t fromLen = sizeof(sockaddr_t); +        int recvlen = recvfrom(sock->Backend, (char*)recvbuf, 1024, 0, &fromAddr, &fromLen); +        if (recvlen < 1) continue; + +        if (fromAddr.sa_family != AF_INET) continue; +        struct sockaddr_in* fromAddrIn = (struct sockaddr_in*)&fromAddr; +        if (memcmp(&fromAddrIn->sin_addr, sock->DestIP, 4)) continue; +        if (ntohs(fromAddrIn->sin_port) != sock->DestPort) continue; + +        printf("UDP: socket %d receiving %d bytes\n", i, recvlen); +        UDP_BuildIncomingFrame(sock, recvbuf, recvlen); +    } + +    return ret; +} + +} diff --git a/src/frontend/qt_sdl/LAN_Socket.h b/src/frontend/qt_sdl/LAN_Socket.h new file mode 100644 index 0000000..8453a5f --- /dev/null +++ b/src/frontend/qt_sdl/LAN_Socket.h @@ -0,0 +1,38 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef LAN_SOCKET_H +#define LAN_SOCKET_H + +#include "../types.h" + +namespace LAN_Socket +{ + +// + + +bool Init(); +void DeInit(); + +int SendPacket(u8* data, int len); +int RecvPacket(u8* data); + +} + +#endif // LAN_SOCKET_H diff --git a/src/frontend/qt_sdl/OSD.cpp b/src/frontend/qt_sdl/OSD.cpp new file mode 100644 index 0000000..4e4e40f --- /dev/null +++ b/src/frontend/qt_sdl/OSD.cpp @@ -0,0 +1,474 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <stdio.h> +#include <string.h> +#include <deque> +#include <SDL2/SDL.h> +#include "../types.h" + +#include "main.h" +#include <QPainter> + +#include "OSD.h" +#include "OSD_shaders.h" +#include "font.h" + +#include "PlatformConfig.h" + +extern MainWindow* mainWindow; + +namespace OSD +{ + +const u32 kOSDMargin = 6; + +struct Item +{ +    Uint32 Timestamp; +    char Text[256]; +    u32 Color; + +    u32 Width, Height; +    u32* Bitmap; + +    bool NativeBitmapLoaded; +    QImage NativeBitmap; + +    bool GLTextureLoaded; +    GLuint GLTexture; + +}; + +std::deque<Item> ItemQueue; + +QOpenGLShaderProgram* Shader; +GLint uScreenSize, uOSDPos, uOSDSize; +GLuint OSDVertexArray; +GLuint OSDVertexBuffer; + +volatile bool Rendering; + + +bool Init(QOpenGLFunctions_3_2_Core* f) +{ +    if (f) +    { +        Shader = new QOpenGLShaderProgram(); +        Shader->addShaderFromSourceCode(QOpenGLShader::Vertex, kScreenVS_OSD); +        Shader->addShaderFromSourceCode(QOpenGLShader::Fragment, kScreenFS_OSD); + +        GLuint pid = Shader->programId(); +        f->glBindAttribLocation(pid, 0, "vPosition"); +        f->glBindFragDataLocation(pid, 0, "oColor"); + +        Shader->link(); + +        Shader->bind(); +        Shader->setUniformValue("OSDTex", (GLint)0); +        Shader->release(); + +        uScreenSize = Shader->uniformLocation("uScreenSize"); +        uOSDPos = Shader->uniformLocation("uOSDPos"); +        uOSDSize = Shader->uniformLocation("uOSDSize"); + +        float vertices[6*2] = +        { +            0, 0, +            1, 1, +            1, 0, +            0, 0, +            0, 1, +            1, 1 +        }; + +        f->glGenBuffers(1, &OSDVertexBuffer); +        f->glBindBuffer(GL_ARRAY_BUFFER, OSDVertexBuffer); +        f->glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW); + +        f->glGenVertexArrays(1, &OSDVertexArray); +        f->glBindVertexArray(OSDVertexArray); +        f->glEnableVertexAttribArray(0); // position +        f->glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 0, (void*)(0)); +    } + +    return true; +} + +void DeInit(QOpenGLFunctions_3_2_Core* f) +{ +    for (auto it = ItemQueue.begin(); it != ItemQueue.end(); ) +    { +        Item& item = *it; + +        if (item.GLTextureLoaded && f) f->glDeleteTextures(1, &item.GLTexture); +        if (item.Bitmap) delete[] item.Bitmap; + +        it = ItemQueue.erase(it); +    } + +    if (f) delete Shader; +} + + +int FindBreakPoint(const char* text, int i) +{ +    // i = character that went out of bounds + +    for (int j = i; j >= 0; j--) +    { +        if (text[j] == ' ') +            return j; +    } + +    return i; +} + +void LayoutText(const char* text, u32* width, u32* height, int* breaks) +{ +    u32 w = 0; +    u32 h = 14; +    u32 totalw = 0; +    u32 maxw = mainWindow->panel->width() - (kOSDMargin*2); +    int lastbreak = -1; +    int numbrk = 0; +    u16* ptr; + +    memset(breaks, 0, sizeof(int)*64); + +    for (int i = 0; text[i] != '\0'; ) +	{ +	    int glyphsize; +		if (text[i] == ' ') +		{ +			glyphsize = 6; +		} +		else +        { +            u32 ch = text[i]; +            if (ch < 0x10 || ch > 0x7E) ch = 0x7F; + +            ptr = &font[(ch-0x10) << 4]; +            glyphsize = ptr[0]; +            if (!glyphsize) glyphsize = 6; +            else            glyphsize += 2; // space around the character +        } + +		w += glyphsize; +		if (w > maxw) +        { +            // wrap shit as needed +            if (text[i] == ' ') +            { +                if (numbrk >= 64) break; +                breaks[numbrk++] = i; +                i++; +            } +            else +            { +                int brk = FindBreakPoint(text, i); +                if (brk != lastbreak) i = brk; + +                if (numbrk >= 64) break; +                breaks[numbrk++] = i; + +                lastbreak = brk; +            } + +            w = 0; +            h += 14; +        } +        else +            i++; + +        if (w > totalw) totalw = w; +    } + +    *width = totalw; +    *height = h; +} + +u32 RainbowColor(u32 inc) +{ +    // inspired from Acmlmboard + +    if      (inc < 100) return 0xFFFF9B9B + (inc << 8); +    else if (inc < 200) return 0xFFFFFF9B - ((inc-100) << 16); +    else if (inc < 300) return 0xFF9BFF9B + (inc-200); +    else if (inc < 400) return 0xFF9BFFFF - ((inc-300) << 8); +    else if (inc < 500) return 0xFF9B9BFF + ((inc-400) << 16); +    else                return 0xFFFF9BFF - (inc-500); +} + +void RenderText(u32 color, const char* text, Item* item) +{ +    u32 w, h; +    int breaks[64]; + +    bool rainbow = (color == 0); +    u32 rainbowinc = ((text[0] * 17) + (SDL_GetTicks() * 13)) % 600; + +    color |= 0xFF000000; +    const u32 shadow = 0xE0000000; + +    LayoutText(text, &w, &h, breaks); + +    item->Width = w; +    item->Height = h; +    item->Bitmap = new u32[w*h]; +    memset(item->Bitmap, 0, w*h*sizeof(u32)); + +    u32 x = 0, y = 1; +    u32 maxw = mainWindow->panel->width() - (kOSDMargin*2); +    int curline = 0; +    u16* ptr; + +    for (int i = 0; text[i] != '\0'; ) +	{ +	    int glyphsize; +		if (text[i] == ' ') +		{ +			x += 6; +		} +		else +        { +            u32 ch = text[i]; +            if (ch < 0x10 || ch > 0x7E) ch = 0x7F; + +            ptr = &font[(ch-0x10) << 4]; +            int glyphsize = ptr[0]; +            if (!glyphsize) x += 6; +            else +            { +                x++; + +                if (rainbow) +                { +                    color = RainbowColor(rainbowinc); +                    rainbowinc = (rainbowinc + 30) % 600; +                } + +                // draw character +                for (int cy = 0; cy < 12; cy++) +                { +                    u16 val = ptr[4+cy]; + +                    for (int cx = 0; cx < glyphsize; cx++) +                    { +                        if (val & (1<<cx)) +                            item->Bitmap[((y+cy) * w) + x+cx] = color; +                    } +                } + +                x += glyphsize; +                x++; +            } +        } + +		i++; +		if (breaks[curline] && i >= breaks[curline]) +        { +            i = breaks[curline++]; +            if (text[i] == ' ') i++; + +            x = 0; +            y += 14; +        } +    } + +    // shadow +    for (y = 0; y < h; y++) +    { +        for (x = 0; x < w; x++) +        { +            u32 val; + +            val = item->Bitmap[(y * w) + x]; +            if ((val >> 24) == 0xFF) continue; + +            if (x > 0)   val  = item->Bitmap[(y * w) + x-1]; +            if (x < w-1) val |= item->Bitmap[(y * w) + x+1]; +            if (y > 0) +            { +                if (x > 0)   val |= item->Bitmap[((y-1) * w) + x-1]; +                val |= item->Bitmap[((y-1) * w) + x]; +                if (x < w-1) val |= item->Bitmap[((y-1) * w) + x+1]; +            } +            if (y < h-1) +            { +                if (x > 0)   val |= item->Bitmap[((y+1) * w) + x-1]; +                val |= item->Bitmap[((y+1) * w) + x]; +                if (x < w-1) val |= item->Bitmap[((y+1) * w) + x+1]; +            } + +            if ((val >> 24) == 0xFF) +                item->Bitmap[(y * w) + x] = shadow; +        } +    } +} + + +void AddMessage(u32 color, const char* text) +{ +    if (!Config::ShowOSD) return; + +    while (Rendering); + +    Item item; + +    item.Timestamp = SDL_GetTicks(); +    strncpy(item.Text, text, 255); item.Text[255] = '\0'; +    item.Color = color; +    item.Bitmap = nullptr; + +    item.NativeBitmapLoaded = false; +    item.GLTextureLoaded = false; + +    ItemQueue.push_back(item); +} + +void Update(QOpenGLFunctions_3_2_Core* f) +{ +    if (!Config::ShowOSD) +    { +        Rendering = true; +        for (auto it = ItemQueue.begin(); it != ItemQueue.end(); ) +        { +            Item& item = *it; + +            if (item.GLTextureLoaded && f) f->glDeleteTextures(1, &item.GLTexture); +            if (item.Bitmap) delete[] item.Bitmap; + +            it = ItemQueue.erase(it); +        } +        Rendering = false; +        return; +    } + +    Rendering = true; + +    Uint32 tick_now = SDL_GetTicks(); +    Uint32 tick_min = tick_now - 2500; + +    for (auto it = ItemQueue.begin(); it != ItemQueue.end(); ) +    { +        Item& item = *it; + +        if (item.Timestamp < tick_min) +        { +            if (item.GLTextureLoaded) f->glDeleteTextures(1, &item.GLTexture); +            if (item.Bitmap) delete[] item.Bitmap; + +            it = ItemQueue.erase(it); +            continue; +        } + +        if (!item.Bitmap) +        { +            RenderText(item.Color, item.Text, &item); +        } + +        it++; +    } + +    Rendering = false; +} + +void DrawNative(QPainter& painter) +{ +    if (!Config::ShowOSD) return; + +    Rendering = true; + +    u32 y = kOSDMargin; + +    painter.resetTransform(); + +    for (auto it = ItemQueue.begin(); it != ItemQueue.end(); ) +    { +        Item& item = *it; + +        if (!item.NativeBitmapLoaded) +        { +            item.NativeBitmap = QImage((const uchar*)item.Bitmap, item.Width, item.Height, QImage::Format_ARGB32_Premultiplied); +            item.NativeBitmapLoaded = true; +        } + +        painter.drawImage(kOSDMargin, y, item.NativeBitmap); + +        y += item.Height; +        it++; +    } + +    Rendering = false; +} + +void DrawGL(QOpenGLFunctions_3_2_Core* f, float w, float h) +{ +    if (!Config::ShowOSD) return; +    if (!mainWindow || !mainWindow->panel) return; + +    Rendering = true; + +    u32 y = kOSDMargin; + +    Shader->bind(); + +    f->glUniform2f(uScreenSize, w, h); + +    f->glBindBuffer(GL_ARRAY_BUFFER, OSDVertexBuffer); +    f->glBindVertexArray(OSDVertexArray); + +    f->glActiveTexture(GL_TEXTURE0); + +    f->glEnable(GL_BLEND); +    f->glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA); + +    for (auto it = ItemQueue.begin(); it != ItemQueue.end(); ) +    { +        Item& item = *it; + +        if (!item.GLTextureLoaded) +        { +            f->glGenTextures(1, &item.GLTexture); +            f->glBindTexture(GL_TEXTURE_2D, item.GLTexture); +            f->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); +            f->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); +            f->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); +            f->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); +            f->glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, item.Width, item.Height, 0, GL_RGBA, GL_UNSIGNED_BYTE, item.Bitmap); + +            item.GLTextureLoaded = true; +        } + +        f->glBindTexture(GL_TEXTURE_2D, item.GLTexture); +        f->glUniform2i(uOSDPos, kOSDMargin, y); +        f->glUniform2i(uOSDSize, item.Width, item.Height); +        f->glDrawArrays(GL_TRIANGLES, 0, 2*3); + +        y += item.Height; +        it++; +    } + +    f->glDisable(GL_BLEND); +    Shader->release(); + +    Rendering = false; +} + +} diff --git a/src/frontend/qt_sdl/OSD.h b/src/frontend/qt_sdl/OSD.h new file mode 100644 index 0000000..79d9df0 --- /dev/null +++ b/src/frontend/qt_sdl/OSD.h @@ -0,0 +1,36 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef OSD_H +#define OSD_H + +namespace OSD +{ + +bool Init(QOpenGLFunctions_3_2_Core* f); +void DeInit(QOpenGLFunctions_3_2_Core* f); + +void AddMessage(u32 color, const char* text); + +void Update(QOpenGLFunctions_3_2_Core* f); +void DrawNative(QPainter& painter); +void DrawGL(QOpenGLFunctions_3_2_Core* f, float w, float h); + +} + +#endif // OSD_H diff --git a/src/frontend/qt_sdl/OSD_shaders.h b/src/frontend/qt_sdl/OSD_shaders.h new file mode 100644 index 0000000..5a64f66 --- /dev/null +++ b/src/frontend/qt_sdl/OSD_shaders.h @@ -0,0 +1,65 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef OSD_SHADERS_H +#define OSD_SHADERS_H + +const char* kScreenVS_OSD = R"(#version 140 + +uniform vec2 uScreenSize; + +uniform ivec2 uOSDPos; +uniform ivec2 uOSDSize; + +in vec2 vPosition; + +smooth out vec2 fTexcoord; + +void main() +{ +    vec4 fpos; + +    vec2 osdpos = (vPosition * vec2(uOSDSize)); +    fTexcoord = osdpos; +    osdpos += uOSDPos; + +    fpos.xy = ((osdpos * 2.0) / uScreenSize) - 1.0; +    fpos.y *= -1; +    fpos.z = 0.0; +    fpos.w = 1.0; + +    gl_Position = fpos; +} +)"; + +const char* kScreenFS_OSD = R"(#version 140 + +uniform sampler2D OSDTex; + +smooth in vec2 fTexcoord; + +out vec4 oColor; + +void main() +{ +    vec4 pixel = texelFetch(OSDTex, ivec2(fTexcoord), 0); +    oColor = pixel.bgra; +} +)"; + +#endif // OSD_SHADERS_H diff --git a/src/frontend/qt_sdl/Platform.cpp b/src/frontend/qt_sdl/Platform.cpp new file mode 100644 index 0000000..edc8d45 --- /dev/null +++ b/src/frontend/qt_sdl/Platform.cpp @@ -0,0 +1,604 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <SDL2/SDL.h> +#include "Platform.h" +#include "PlatformConfig.h" +#include "LAN_Socket.h" +#include "LAN_PCap.h" +#include <string> + +#ifdef __WIN32__ +    #define NTDDI_VERSION		0x06000000 // GROSS FUCKING HACK +    #include <windows.h> +    //#include <knownfolders.h> // FUCK THAT SHIT +    extern "C" const GUID DECLSPEC_SELECTANY FOLDERID_RoamingAppData = {0x3eb685db, 0x65f9, 0x4cf6, {0xa0, 0x3a, 0xe3, 0xef, 0x65, 0x72, 0x9f, 0x3d}}; +    #include <shlobj.h> +	#include <winsock2.h> +	#include <ws2tcpip.h> +	#define socket_t    SOCKET +	#define sockaddr_t  SOCKADDR +#else +    #include <glib.h> +	#include <unistd.h> +	#include <arpa/inet.h> +	#include <netinet/in.h> +	#include <sys/select.h> +	#include <sys/socket.h> +	#define socket_t    int +	#define sockaddr_t  struct sockaddr +	#define closesocket close +#endif + +#ifndef INVALID_SOCKET +#define INVALID_SOCKET  (socket_t)-1 +#endif + + +char* EmuDirectory; + +void emuStop(); +void* oglGetProcAddress(const char* proc); + + +namespace Platform +{ + + +typedef struct +{ +    SDL_Thread* ID; +    void (*Func)(); + +} ThreadData; + +int ThreadEntry(void* data) +{ +    ThreadData* thread = (ThreadData*)data; +    thread->Func(); +    return 0; +} + + +socket_t MPSocket; +sockaddr_t MPSendAddr; +u8 PacketBuffer[2048]; + +#define NIFI_VER 1 + + +void Init(int argc, char** argv) +{ +#if defined(__WIN32__) || defined(UNIX_PORTABLE) +    if (argc > 0 && strlen(argv[0]) > 0) +    { +        int len = strlen(argv[0]); +        while (len > 0) +        { +            if (argv[0][len] == '/') break; +            if (argv[0][len] == '\\') break; +            len--; +        } +        if (len > 0) +        { +            EmuDirectory = new char[len+1]; +            strncpy(EmuDirectory, argv[0], len); +            EmuDirectory[len] = '\0'; +        } +        else +        { +            EmuDirectory = new char[2]; +            strcpy(EmuDirectory, "."); +        } +    } +    else +    { +        EmuDirectory = new char[2]; +        strcpy(EmuDirectory, "."); +    } +#else +	const char* confdir = g_get_user_config_dir(); +	const char* confname = "/melonDS"; +	int cdlen = strlen(confdir); +	int cnlen = strlen(confname); +	EmuDirectory = new char[cdlen + cnlen + 1]; +	strncpy(&EmuDirectory[0], confdir, cdlen); +	strncpy(&EmuDirectory[cdlen], confname, cnlen); +	EmuDirectory[cdlen+cnlen] = '\0'; +#endif +} + +void DeInit() +{ +    delete[] EmuDirectory; +} + + +void StopEmu() +{ +    emuStop(); +} + + +FILE* OpenFile(const char* path, const char* mode, bool mustexist) +{ +    FILE* ret; + +#ifdef __WIN32__ + +    int len = MultiByteToWideChar(CP_UTF8, 0, path, -1, NULL, 0); +    if (len < 1) return NULL; +    WCHAR* fatpath = new WCHAR[len]; +    int res = MultiByteToWideChar(CP_UTF8, 0, path, -1, fatpath, len); +    if (res != len) { delete[] fatpath; return NULL; } // checkme? + +    // this will be more than enough +    WCHAR fatmode[4]; +    fatmode[0] = mode[0]; +    fatmode[1] = mode[1]; +    fatmode[2] = mode[2]; +    fatmode[3] = 0; + +    if (mustexist) +    { +        ret = _wfopen(fatpath, L"rb"); +        if (ret) ret = _wfreopen(fatpath, fatmode, ret); +    } +    else +        ret = _wfopen(fatpath, fatmode); + +    delete[] fatpath; + +#else + +    if (mustexist) +    { +        ret = fopen(path, "rb"); +        if (ret) ret = freopen(path, mode, ret); +    } +    else +        ret = fopen(path, mode); + +#endif + +    return ret; +} + +#if !defined(UNIX_PORTABLE) && !defined(__WIN32__) + +FILE* OpenLocalFile(const char* path, const char* mode) +{ +    std::string fullpath; +    if (path[0] == '/') +    { +        // If it's an absolute path, just open that. +        fullpath = std::string(path); +    } +    else +    { +        // Check user configuration directory +        std::string confpath = std::string(g_get_user_config_dir()) + "/melonDS/"; +        g_mkdir_with_parents(confpath.c_str(), 0755); +        fullpath = confpath + path; +    } + +    return OpenFile(fullpath.c_str(), mode, mode[0] != 'w'); +} + +FILE* OpenDataFile(const char* path) +{ +    const char* melondir = "melonDS"; +    const char* const* sys_dirs = g_get_system_data_dirs(); +    const char* user_dir = g_get_user_data_dir(); + +    // First check the user's data directory +    char* fullpath = g_build_path("/", user_dir, melondir, path, NULL); +    if (access(fullpath, R_OK) == 0) +    { +        FILE* f = fopen(fullpath, "r"); +        g_free(fullpath); +        return f; +    } +    free(fullpath); + +    // Then check the system data directories +    for (size_t i = 0; sys_dirs[i] != NULL; i++) +    { +        const char* dir = sys_dirs[i]; +        char* fullpath = g_build_path("/", dir, melondir, path, NULL); + +        if (access(fullpath, R_OK) == 0) +        { +            FILE* f = fopen(fullpath, "r"); +            g_free(fullpath); +            return f; +        } +        free(fullpath); +    } + +	FILE* f = fopen(path, "rb"); +	if (f) return f; + +    return NULL; +} + +#else + +FILE* OpenLocalFile(const char* path, const char* mode) +{ +    bool relpath = false; +    int pathlen = strlen(path); + +#ifdef __WIN32__ +    if (pathlen > 3) +    { +        if (path[1] == ':' && (path[2] == '\\' || path[2] == '/')) +            return OpenFile(path, mode); +    } +#else +    if (pathlen > 1) +    { +        if (path[0] == '/') +            return OpenFile(path, mode); +    } +#endif + +    if (pathlen >= 3) +    { +        if (path[0] == '.' && path[1] == '.' && (path[2] == '/' || path[2] == '\\')) +            relpath = true; +    } + +    int emudirlen = strlen(EmuDirectory); +    char* emudirpath; +    if (emudirlen) +    { +        int len = emudirlen + 1 + pathlen + 1; +        emudirpath = new char[len]; +        strncpy(&emudirpath[0], EmuDirectory, emudirlen); +        emudirpath[emudirlen] = '/'; +        strncpy(&emudirpath[emudirlen+1], path, pathlen); +        emudirpath[emudirlen+1+pathlen] = '\0'; +    } +    else +    { +        emudirpath = new char[pathlen+1]; +        strncpy(&emudirpath[0], path, pathlen); +        emudirpath[pathlen] = '\0'; +    } + +    // Locations are application directory, and AppData/melonDS on Windows or XDG_CONFIG_HOME/melonDS on Linux + +    FILE* f; + +    // First check current working directory +    f = OpenFile(path, mode, true); +    if (f) { delete[] emudirpath; return f; } + +    // then emu directory +    f = OpenFile(emudirpath, mode, true); +    if (f) { delete[] emudirpath; return f; } + +#ifdef __WIN32__ + +    // a path relative to AppData wouldn't make much sense +    if (!relpath) +    { +        // Now check AppData +        PWSTR appDataPath = NULL; +        SHGetKnownFolderPath(FOLDERID_RoamingAppData, 0, NULL, &appDataPath); +        if (!appDataPath) +        { +            delete[] emudirpath; +            return NULL; +        } + +        // this will be more than enough +        WCHAR fatperm[4]; +        fatperm[0] = mode[0]; +        fatperm[1] = mode[1]; +        fatperm[2] = mode[2]; +        fatperm[3] = 0; + +        int fnlen = MultiByteToWideChar(CP_UTF8, 0, path, -1, NULL, 0); +        if (fnlen < 1) { delete[] emudirpath; return NULL; } +        WCHAR* wfileName = new WCHAR[fnlen]; +        int res = MultiByteToWideChar(CP_UTF8, 0, path, -1, wfileName, fnlen); +        if (res != fnlen) { delete[] wfileName; delete[] emudirpath; return NULL; } // checkme? + +        const WCHAR* appdir = L"\\melonDS\\"; + +        int pos = wcslen(appDataPath); +        void* ptr = CoTaskMemRealloc(appDataPath, (pos+wcslen(appdir)+fnlen+1)*sizeof(WCHAR)); +        if (!ptr) { delete[] wfileName; delete[] emudirpath; return NULL; } // oh well +        appDataPath = (PWSTR)ptr; + +        wcscpy(&appDataPath[pos], appdir); pos += wcslen(appdir); +        wcscpy(&appDataPath[pos], wfileName); + +        f = _wfopen(appDataPath, L"rb"); +        if (f) f = _wfreopen(appDataPath, fatperm, f); +        CoTaskMemFree(appDataPath); +        delete[] wfileName; +        if (f) { delete[] emudirpath; return f; } +    } + +#else + +    if (!relpath) +    { +        // Now check XDG_CONFIG_HOME +        // TODO: check for memory leak there +        std::string fullpath = std::string(g_get_user_config_dir()) + "/melonDS/" + path; +        f = OpenFile(fullpath.c_str(), mode, true); +        if (f) { delete[] emudirpath; return f; } +    } + +#endif + +    if (mode[0] != 'r') +    { +        f = OpenFile(emudirpath, mode); +        if (f) { delete[] emudirpath; return f; } +    } + +    delete[] emudirpath; +    return NULL; +} + +FILE* OpenDataFile(const char* path) +{ +	return OpenLocalFile(path, "rb"); +} + +#endif + + +void* Thread_Create(void (*func)()) +{ +    ThreadData* data = new ThreadData; +    data->Func = func; +    data->ID = SDL_CreateThread(ThreadEntry, "melonDS core thread", data); +    return data; +} + +void Thread_Free(void* thread) +{ +    delete (ThreadData*)thread; +} + +void Thread_Wait(void* thread) +{ +    SDL_WaitThread((SDL_Thread*)((ThreadData*)thread)->ID, NULL); +} + + +void* Semaphore_Create() +{ +    return SDL_CreateSemaphore(0); +} + +void Semaphore_Free(void* sema) +{ +    SDL_DestroySemaphore((SDL_sem*)sema); +} + +void Semaphore_Reset(void* sema) +{ +    while (SDL_SemTryWait((SDL_sem*)sema) == 0); +} + +void Semaphore_Wait(void* sema) +{ +    SDL_SemWait((SDL_sem*)sema); +} + +void Semaphore_Post(void* sema) +{ +    SDL_SemPost((SDL_sem*)sema); +} + + +void* GL_GetProcAddress(const char* proc) +{ +    return oglGetProcAddress(proc); +} + + +bool MP_Init() +{ +    int opt_true = 1; +    int res; + +#ifdef __WIN32__ +    WSADATA wsadata; +    if (WSAStartup(MAKEWORD(2, 2), &wsadata) != 0) +    { +        return false; +    } +#endif // __WIN32__ + +    MPSocket = socket(AF_INET, SOCK_DGRAM, 0); +	if (MPSocket < 0) +	{ +		return false; +	} + +	res = setsockopt(MPSocket, SOL_SOCKET, SO_REUSEADDR, (const char*)&opt_true, sizeof(int)); +	if (res < 0) +	{ +		closesocket(MPSocket); +		MPSocket = INVALID_SOCKET; +		return false; +	} + +	sockaddr_t saddr; +	saddr.sa_family = AF_INET; +	*(u32*)&saddr.sa_data[2] = htonl(Config::SocketBindAnyAddr ? INADDR_ANY : INADDR_LOOPBACK); +	*(u16*)&saddr.sa_data[0] = htons(7064); +	res = bind(MPSocket, &saddr, sizeof(sockaddr_t)); +	if (res < 0) +	{ +		closesocket(MPSocket); +		MPSocket = INVALID_SOCKET; +		return false; +	} + +	res = setsockopt(MPSocket, SOL_SOCKET, SO_BROADCAST, (const char*)&opt_true, sizeof(int)); +	if (res < 0) +	{ +		closesocket(MPSocket); +		MPSocket = INVALID_SOCKET; +		return false; +	} + +	MPSendAddr.sa_family = AF_INET; +	*(u32*)&MPSendAddr.sa_data[2] = htonl(INADDR_BROADCAST); +	*(u16*)&MPSendAddr.sa_data[0] = htons(7064); + +	return true; +} + +void MP_DeInit() +{ +    if (MPSocket >= 0) +        closesocket(MPSocket); + +#ifdef __WIN32__ +    WSACleanup(); +#endif // __WIN32__ +} + +int MP_SendPacket(u8* data, int len) +{ +    if (MPSocket < 0) +        return 0; + +    if (len > 2048-8) +    { +        printf("MP_SendPacket: error: packet too long (%d)\n", len); +        return 0; +    } + +    *(u32*)&PacketBuffer[0] = htonl(0x4946494E); // NIFI +    PacketBuffer[4] = NIFI_VER; +    PacketBuffer[5] = 0; +    *(u16*)&PacketBuffer[6] = htons(len); +    memcpy(&PacketBuffer[8], data, len); + +    int slen = sendto(MPSocket, (const char*)PacketBuffer, len+8, 0, &MPSendAddr, sizeof(sockaddr_t)); +    if (slen < 8) return 0; +    return slen - 8; +} + +int MP_RecvPacket(u8* data, bool block) +{ +    if (MPSocket < 0) +        return 0; + +    fd_set fd; +	struct timeval tv; + +	FD_ZERO(&fd); +	FD_SET(MPSocket, &fd); +	tv.tv_sec = 0; +	tv.tv_usec = block ? 5000 : 0; + +	if (!select(MPSocket+1, &fd, 0, 0, &tv)) +    { +        return 0; +    } + +    sockaddr_t fromAddr; +    socklen_t fromLen = sizeof(sockaddr_t); +    int rlen = recvfrom(MPSocket, (char*)PacketBuffer, 2048, 0, &fromAddr, &fromLen); +    if (rlen < 8+24) +    { +        return 0; +    } +    rlen -= 8; + +    if (ntohl(*(u32*)&PacketBuffer[0]) != 0x4946494E) +    { +        return 0; +    } + +    if (PacketBuffer[4] != NIFI_VER) +    { +        return 0; +    } + +    if (ntohs(*(u16*)&PacketBuffer[6]) != rlen) +    { +        return 0; +    } + +    memcpy(data, &PacketBuffer[8], rlen); +    return rlen; +} + + + +bool LAN_Init() +{ +    if (Config::DirectLAN) +    { +        if (!LAN_PCap::Init(true)) +            return false; +    } +    else +    { +        if (!LAN_Socket::Init()) +            return false; +    } + +    return true; +} + +void LAN_DeInit() +{ +    // checkme. blarg +    //if (Config::DirectLAN) +    //    LAN_PCap::DeInit(); +    //else +    //    LAN_Socket::DeInit(); +    LAN_PCap::DeInit(); +    LAN_Socket::DeInit(); +} + +int LAN_SendPacket(u8* data, int len) +{ +    if (Config::DirectLAN) +        return LAN_PCap::SendPacket(data, len); +    else +        return LAN_Socket::SendPacket(data, len); +} + +int LAN_RecvPacket(u8* data) +{ +    if (Config::DirectLAN) +        return LAN_PCap::RecvPacket(data); +    else +        return LAN_Socket::RecvPacket(data); +} + + +} diff --git a/src/frontend/qt_sdl/PlatformConfig.cpp b/src/frontend/qt_sdl/PlatformConfig.cpp new file mode 100644 index 0000000..03fd2ac --- /dev/null +++ b/src/frontend/qt_sdl/PlatformConfig.cpp @@ -0,0 +1,153 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <stdio.h> +#include <string.h> +#include <stdlib.h> +#include "PlatformConfig.h" + +namespace Config +{ + +int KeyMapping[12]; +int JoyMapping[12]; + +int HKKeyMapping[HK_MAX]; +int HKJoyMapping[HK_MAX]; + +int JoystickID; + +int WindowWidth; +int WindowHeight; +int WindowMaximized; + +int ScreenRotation; +int ScreenGap; +int ScreenLayout; +int ScreenSizing; +int IntegerScaling; +int ScreenFilter; + +int ScreenUseGL; +int ScreenVSync; +int ScreenVSyncInterval; + +int LimitFPS; +int AudioSync; +int ShowOSD; + +int DirectBoot; + +int SocketBindAnyAddr; +char LANDevice[128]; +int DirectLAN; + +int SavestateRelocSRAM; + +int AudioVolume; +int MicInputType; +char MicWavPath[1024]; + +char LastROMFolder[1024]; + + +ConfigEntry PlatformConfigFile[] = +{ +    {"Key_A",      0, &KeyMapping[0],  -1, NULL, 0}, +    {"Key_B",      0, &KeyMapping[1],  -1, NULL, 0}, +    {"Key_Select", 0, &KeyMapping[2],  -1, NULL, 0}, +    {"Key_Start",  0, &KeyMapping[3],  -1, NULL, 0}, +    {"Key_Right",  0, &KeyMapping[4],  -1, NULL, 0}, +    {"Key_Left",   0, &KeyMapping[5],  -1, NULL, 0}, +    {"Key_Up",     0, &KeyMapping[6],  -1, NULL, 0}, +    {"Key_Down",   0, &KeyMapping[7],  -1, NULL, 0}, +    {"Key_R",      0, &KeyMapping[8],  -1, NULL, 0}, +    {"Key_L",      0, &KeyMapping[9],  -1, NULL, 0}, +    {"Key_X",      0, &KeyMapping[10], -1, NULL, 0}, +    {"Key_Y",      0, &KeyMapping[11], -1, NULL, 0}, + +    {"Joy_A",      0, &JoyMapping[0],  -1, NULL, 0}, +    {"Joy_B",      0, &JoyMapping[1],  -1, NULL, 0}, +    {"Joy_Select", 0, &JoyMapping[2],  -1, NULL, 0}, +    {"Joy_Start",  0, &JoyMapping[3],  -1, NULL, 0}, +    {"Joy_Right",  0, &JoyMapping[4],  -1, NULL, 0}, +    {"Joy_Left",   0, &JoyMapping[5],  -1, NULL, 0}, +    {"Joy_Up",     0, &JoyMapping[6],  -1, NULL, 0}, +    {"Joy_Down",   0, &JoyMapping[7],  -1, NULL, 0}, +    {"Joy_R",      0, &JoyMapping[8],  -1, NULL, 0}, +    {"Joy_L",      0, &JoyMapping[9],  -1, NULL, 0}, +    {"Joy_X",      0, &JoyMapping[10], -1, NULL, 0}, +    {"Joy_Y",      0, &JoyMapping[11], -1, NULL, 0}, + +    {"HKKey_Lid",                 0, &HKKeyMapping[HK_Lid],                 -1, NULL, 0}, +    {"HKKey_Mic",                 0, &HKKeyMapping[HK_Mic],                 -1, NULL, 0}, +    {"HKKey_Pause",               0, &HKKeyMapping[HK_Pause],               -1, NULL, 0}, +    {"HKKey_Reset",               0, &HKKeyMapping[HK_Reset],               -1, NULL, 0}, +    {"HKKey_FastForward",         0, &HKKeyMapping[HK_FastForward],         -1, NULL, 0}, +    {"HKKey_FastForwardToggle",   0, &HKKeyMapping[HK_FastForwardToggle],   -1, NULL, 0}, +    {"HKKey_SolarSensorDecrease", 0, &HKKeyMapping[HK_SolarSensorDecrease], -1, NULL, 0}, +    {"HKKey_SolarSensorIncrease", 0, &HKKeyMapping[HK_SolarSensorIncrease], -1, NULL, 0}, + +    {"HKJoy_Lid",                 0, &HKJoyMapping[HK_Lid],                 -1, NULL, 0}, +    {"HKJoy_Mic",                 0, &HKJoyMapping[HK_Mic],                 -1, NULL, 0}, +    {"HKJoy_Pause",               0, &HKJoyMapping[HK_Pause],               -1, NULL, 0}, +    {"HKJoy_Reset",               0, &HKJoyMapping[HK_Reset],               -1, NULL, 0}, +    {"HKJoy_FastForward",         0, &HKJoyMapping[HK_FastForward],         -1, NULL, 0}, +    {"HKJoy_FastForwardToggle",   0, &HKJoyMapping[HK_FastForwardToggle],   -1, NULL, 0}, +    {"HKJoy_SolarSensorDecrease", 0, &HKJoyMapping[HK_SolarSensorDecrease], -1, NULL, 0}, +    {"HKJoy_SolarSensorIncrease", 0, &HKJoyMapping[HK_SolarSensorIncrease], -1, NULL, 0}, + +    {"JoystickID", 0, &JoystickID, 0, NULL, 0}, + +    {"WindowWidth",  0, &WindowWidth,  256, NULL, 0}, +    {"WindowHeight", 0, &WindowHeight, 384, NULL, 0}, +    {"WindowMax",    0, &WindowMaximized, 0, NULL, 0}, + +    {"ScreenRotation", 0, &ScreenRotation, 0, NULL, 0}, +    {"ScreenGap",      0, &ScreenGap,      0, NULL, 0}, +    {"ScreenLayout",   0, &ScreenLayout,   0, NULL, 0}, +    {"ScreenSizing",   0, &ScreenSizing,   0, NULL, 0}, +    {"IntegerScaling", 0, &IntegerScaling, 0, NULL, 0}, +    {"ScreenFilter",   0, &ScreenFilter,   1, NULL, 0}, + +    {"ScreenUseGL",         0, &ScreenUseGL,         1, NULL, 0}, +    {"ScreenVSync",         0, &ScreenVSync,         0, NULL, 0}, +    {"ScreenVSyncInterval", 0, &ScreenVSyncInterval, 1, NULL, 0}, + +    {"LimitFPS", 0, &LimitFPS, 0, NULL, 0}, +    {"AudioSync", 0, &AudioSync, 1, NULL, 0}, +    {"ShowOSD", 0, &ShowOSD, 1, NULL, 0}, + +    {"DirectBoot", 0, &DirectBoot, 1, NULL, 0}, + +    {"SockBindAnyAddr", 0, &SocketBindAnyAddr, 0, NULL, 0}, +    {"LANDevice", 1, LANDevice, 0, "", 127}, +    {"DirectLAN", 0, &DirectLAN, 0, NULL, 0}, + +    {"SavStaRelocSRAM", 0, &SavestateRelocSRAM, 0, NULL, 0}, + +    {"AudioVolume", 0, &AudioVolume, 256, NULL, 0}, +    {"MicInputType", 0, &MicInputType, 1, NULL, 0}, +    {"MicWavPath", 1, MicWavPath, 0, "", 1023}, + +    {"LastROMFolder", 1, LastROMFolder, 0, "", 1023}, + +    {"", -1, NULL, 0, NULL, 0} +}; + +} diff --git a/src/frontend/qt_sdl/PlatformConfig.h b/src/frontend/qt_sdl/PlatformConfig.h new file mode 100644 index 0000000..cc288b6 --- /dev/null +++ b/src/frontend/qt_sdl/PlatformConfig.h @@ -0,0 +1,83 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef PLATFORMCONFIG_H +#define PLATFORMCONFIG_H + +#include "Config.h" + +enum +{ +    HK_Lid = 0, +    HK_Mic, +    HK_Pause, +    HK_Reset, +    HK_FastForward, +    HK_FastForwardToggle, +    HK_SolarSensorDecrease, +    HK_SolarSensorIncrease, +    HK_MAX +}; + +namespace Config +{ + +extern int KeyMapping[12]; +extern int JoyMapping[12]; + +extern int HKKeyMapping[HK_MAX]; +extern int HKJoyMapping[HK_MAX]; + +extern int JoystickID; + +extern int WindowWidth; +extern int WindowHeight; +extern int WindowMaximized; + +extern int ScreenRotation; +extern int ScreenGap; +extern int ScreenLayout; +extern int ScreenSizing; +extern int IntegerScaling; +extern int ScreenFilter; + +extern int ScreenUseGL; +extern int ScreenVSync; +extern int ScreenVSyncInterval; + +extern int LimitFPS; +extern int AudioSync; +extern int ShowOSD; + +extern int DirectBoot; + +extern int SocketBindAnyAddr; +extern char LANDevice[128]; +extern int DirectLAN; + +extern int SavestateRelocSRAM; + +extern int AudioVolume; +extern int MicInputType; +extern char MicWavPath[1024]; + +extern char LastROMFolder[1024]; + +} + +#endif // PLATFORMCONFIG_H diff --git a/src/frontend/qt_sdl/VideoSettingsDialog.cpp b/src/frontend/qt_sdl/VideoSettingsDialog.cpp new file mode 100644 index 0000000..ba433c3 --- /dev/null +++ b/src/frontend/qt_sdl/VideoSettingsDialog.cpp @@ -0,0 +1,169 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <stdio.h> +#include <QFileDialog> + +#include "types.h" +#include "Platform.h" +#include "Config.h" +#include "PlatformConfig.h" + +#include "VideoSettingsDialog.h" +#include "ui_VideoSettingsDialog.h" + + +VideoSettingsDialog* VideoSettingsDialog::currentDlg = nullptr; + + +VideoSettingsDialog::VideoSettingsDialog(QWidget* parent) : QDialog(parent), ui(new Ui::VideoSettingsDialog) +{ +    ui->setupUi(this); +    setAttribute(Qt::WA_DeleteOnClose); + +    oldRenderer = Config::_3DRenderer; +    oldGLDisplay = Config::ScreenUseGL; +    oldVSync = Config::ScreenVSync; +    oldVSyncInterval = Config::ScreenVSyncInterval; +    oldSoftThreaded = Config::Threaded3D; +    oldGLScale = Config::GL_ScaleFactor; + +    grp3DRenderer = new QButtonGroup(this); +    grp3DRenderer->addButton(ui->rb3DSoftware, 0); +    grp3DRenderer->addButton(ui->rb3DOpenGL,   1); +    connect(grp3DRenderer, SIGNAL(buttonClicked(int)), this, SLOT(onChange3DRenderer(int))); +    grp3DRenderer->button(Config::_3DRenderer)->setChecked(true); + +    ui->cbGLDisplay->setChecked(Config::ScreenUseGL != 0); + +    ui->cbVSync->setChecked(Config::ScreenVSync != 0); +    ui->sbVSyncInterval->setValue(Config::ScreenVSyncInterval); + +    ui->cbSoftwareThreaded->setChecked(Config::Threaded3D != 0); + +    for (int i = 1; i <= 16; i++) +        ui->cbxGLResolution->addItem(QString("%1x native (%2x%3)").arg(i).arg(256*i).arg(192*i)); +    ui->cbxGLResolution->setCurrentIndex(Config::GL_ScaleFactor-1); + +    if (!Config::ScreenVSync) +        ui->sbVSyncInterval->setEnabled(false); + +    if (Config::_3DRenderer == 0) +    { +        ui->cbGLDisplay->setEnabled(true); +        ui->cbSoftwareThreaded->setEnabled(true); +        ui->cbxGLResolution->setEnabled(false); +    } +    else +    { +        ui->cbGLDisplay->setEnabled(false); +        ui->cbSoftwareThreaded->setEnabled(false); +        ui->cbxGLResolution->setEnabled(true); +    } +} + +VideoSettingsDialog::~VideoSettingsDialog() +{ +    delete ui; +} + +void VideoSettingsDialog::on_VideoSettingsDialog_accepted() +{ +    Config::Save(); + +    closeDlg(); +} + +void VideoSettingsDialog::on_VideoSettingsDialog_rejected() +{ +    bool old_gl = (Config::ScreenUseGL != 0) || (Config::_3DRenderer != 0); + +    Config::_3DRenderer = oldRenderer; +    Config::ScreenUseGL = oldGLDisplay; +    Config::ScreenVSync = oldVSync; +    Config::ScreenVSyncInterval = oldVSyncInterval; +    Config::Threaded3D = oldSoftThreaded; +    Config::GL_ScaleFactor = oldGLScale; + +    bool new_gl = (Config::ScreenUseGL != 0) || (Config::_3DRenderer != 0); +    emit updateVideoSettings(old_gl != new_gl); + +    closeDlg(); +} + +void VideoSettingsDialog::onChange3DRenderer(int renderer) +{ +    bool old_gl = (Config::ScreenUseGL != 0) || (Config::_3DRenderer != 0); + +    Config::_3DRenderer = renderer; + +    if (renderer == 0) +    { +        ui->cbGLDisplay->setEnabled(true); +        ui->cbSoftwareThreaded->setEnabled(true); +        ui->cbxGLResolution->setEnabled(false); +    } +    else +    { +        ui->cbGLDisplay->setEnabled(false); +        ui->cbSoftwareThreaded->setEnabled(false); +        ui->cbxGLResolution->setEnabled(true); +    } + +    bool new_gl = (Config::ScreenUseGL != 0) || (Config::_3DRenderer != 0); +    emit updateVideoSettings(old_gl != new_gl); +} + +void VideoSettingsDialog::on_cbGLDisplay_stateChanged(int state) +{ +    bool old_gl = (Config::ScreenUseGL != 0) || (Config::_3DRenderer != 0); + +    Config::ScreenUseGL = (state != 0); + +    bool new_gl = (Config::ScreenUseGL != 0) || (Config::_3DRenderer != 0); +    emit updateVideoSettings(old_gl != new_gl); +} + +void VideoSettingsDialog::on_cbVSync_stateChanged(int state) +{ +    bool vsync = (state != 0); +    ui->sbVSyncInterval->setEnabled(vsync); +    Config::ScreenVSync = vsync; +} + +void VideoSettingsDialog::on_sbVSyncInterval_valueChanged(int val) +{ +    Config::ScreenVSyncInterval = val; +} + +void VideoSettingsDialog::on_cbSoftwareThreaded_stateChanged(int state) +{ +    Config::Threaded3D = (state != 0); + +    emit updateVideoSettings(false); +} + +void VideoSettingsDialog::on_cbxGLResolution_currentIndexChanged(int idx) +{ +    // prevent a spurious change +    if (ui->cbxGLResolution->count() < 16) return; + +    Config::GL_ScaleFactor = idx+1; + +    emit updateVideoSettings(false); +} diff --git a/src/frontend/qt_sdl/VideoSettingsDialog.h b/src/frontend/qt_sdl/VideoSettingsDialog.h new file mode 100644 index 0000000..2311d4d --- /dev/null +++ b/src/frontend/qt_sdl/VideoSettingsDialog.h @@ -0,0 +1,84 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef VIDEOSETTINGSDIALOG_H +#define VIDEOSETTINGSDIALOG_H + +#include <QDialog> +#include <QButtonGroup> + +namespace Ui { class VideoSettingsDialog; } +class VideoSettingsDialog; + +class VideoSettingsDialog : public QDialog +{ +    Q_OBJECT + +public: +    explicit VideoSettingsDialog(QWidget* parent); +    ~VideoSettingsDialog(); + +    static VideoSettingsDialog* currentDlg; +    static VideoSettingsDialog* openDlg(QWidget* parent) +    { +        if (currentDlg) +        { +            currentDlg->activateWindow(); +            return currentDlg; +        } + +        currentDlg = new VideoSettingsDialog(parent); +        currentDlg->show(); +        return currentDlg; +    } +    static void closeDlg() +    { +        currentDlg = nullptr; +    } + +signals: +    void updateVideoSettings(bool glchange); + +private slots: +    void on_VideoSettingsDialog_accepted(); +    void on_VideoSettingsDialog_rejected(); + +    void onChange3DRenderer(int renderer); +    void on_cbGLDisplay_stateChanged(int state); +    void on_cbVSync_stateChanged(int state); +    void on_sbVSyncInterval_valueChanged(int val); + +    void on_cbxGLResolution_currentIndexChanged(int idx); + +    void on_cbSoftwareThreaded_stateChanged(int state); + +private: +    Ui::VideoSettingsDialog* ui; + +    QButtonGroup* grp3DRenderer; + +    int oldRenderer; +    int oldGLDisplay; +    int oldVSync; +    int oldVSyncInterval; +    int oldSoftThreaded; +    int oldGLScale; +}; + +#endif // VIDEOSETTINGSDIALOG_H + diff --git a/src/frontend/qt_sdl/VideoSettingsDialog.ui b/src/frontend/qt_sdl/VideoSettingsDialog.ui new file mode 100644 index 0000000..6cdd5d8 --- /dev/null +++ b/src/frontend/qt_sdl/VideoSettingsDialog.ui @@ -0,0 +1,229 @@ +<?xml version="1.0" encoding="UTF-8"?> +<ui version="4.0"> + <class>VideoSettingsDialog</class> + <widget class="QDialog" name="VideoSettingsDialog"> +  <property name="geometry"> +   <rect> +    <x>0</x> +    <y>0</y> +    <width>482</width> +    <height>237</height> +   </rect> +  </property> +  <property name="sizePolicy"> +   <sizepolicy hsizetype="Fixed" vsizetype="Fixed"> +    <horstretch>0</horstretch> +    <verstretch>0</verstretch> +   </sizepolicy> +  </property> +  <property name="windowTitle"> +   <string>Video settings - melonDS</string> +  </property> +  <layout class="QGridLayout" name="gridLayout"> +   <property name="sizeConstraint"> +    <enum>QLayout::SetFixedSize</enum> +   </property> +   <item row="1" column="1"> +    <widget class="QGroupBox" name="groupBox_3"> +     <property name="title"> +      <string>OpenGL renderer</string> +     </property> +     <layout class="QGridLayout" name="gridLayout_4"> +      <item row="0" column="0"> +       <widget class="QLabel" name="label_3"> +        <property name="text"> +         <string>Internal resolution:</string> +        </property> +       </widget> +      </item> +      <item row="1" column="0"> +       <widget class="QComboBox" name="cbxGLResolution"> +        <property name="whatsThis"> +         <string><html><head/><body><p>The resolution at which the 3D graphics will be rendered. Higher resolutions improve graphics quality when the main window is enlarged, but may also cause glitches.</p></body></html></string> +        </property> +       </widget> +      </item> +     </layout> +    </widget> +   </item> +   <item row="0" column="1"> +    <widget class="QGroupBox" name="groupBox_2"> +     <property name="title"> +      <string>Software renderer</string> +     </property> +     <layout class="QGridLayout" name="gridLayout_3"> +      <item row="0" column="0"> +       <widget class="QCheckBox" name="cbSoftwareThreaded"> +        <property name="whatsThis"> +         <string><html><head/><body><p>Run the software renderer on a separate thread. Yields better performance on multi-core CPUs.</p></body></html></string> +        </property> +        <property name="text"> +         <string>Use separate thread</string> +        </property> +       </widget> +      </item> +     </layout> +    </widget> +   </item> +   <item row="4" column="0" colspan="2"> +    <widget class="QDialogButtonBox" name="buttonBox"> +     <property name="orientation"> +      <enum>Qt::Horizontal</enum> +     </property> +     <property name="standardButtons"> +      <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set> +     </property> +    </widget> +   </item> +   <item row="2" column="1"> +    <spacer name="verticalSpacer_2"> +     <property name="orientation"> +      <enum>Qt::Vertical</enum> +     </property> +     <property name="sizeHint" stdset="0"> +      <size> +       <width>20</width> +       <height>40</height> +      </size> +     </property> +    </spacer> +   </item> +   <item row="0" column="0" rowspan="3"> +    <widget class="QGroupBox" name="groupBox"> +     <property name="title"> +      <string>Display settings</string> +     </property> +     <layout class="QGridLayout" name="gridLayout_2"> +      <item row="6" column="0"> +       <widget class="QLabel" name="label_2"> +        <property name="sizePolicy"> +         <sizepolicy hsizetype="Fixed" vsizetype="Fixed"> +          <horstretch>0</horstretch> +          <verstretch>0</verstretch> +         </sizepolicy> +        </property> +        <property name="whatsThis"> +         <string><html><head/><body><p>The interval at which to synchronize to the monitor's refresh rate. Set to 1 for a 60Hz monitor, 2 for 120Hz, ...</p></body></html></string> +        </property> +        <property name="text"> +         <string>VSync interval:</string> +        </property> +       </widget> +      </item> +      <item row="6" column="1"> +       <widget class="QSpinBox" name="sbVSyncInterval"> +        <property name="whatsThis"> +         <string><html><head/><body><p>The interval at which to synchronize to the monitor's refresh rate. Set to 1 for a 60Hz monitor, 2 for 120Hz, ...</p></body></html></string> +        </property> +        <property name="minimum"> +         <number>1</number> +        </property> +        <property name="maximum"> +         <number>20</number> +        </property> +       </widget> +      </item> +      <item row="4" column="0" colspan="2"> +       <widget class="QCheckBox" name="cbGLDisplay"> +        <property name="whatsThis"> +         <string><html><head/><body><p>Use OpenGL to draw the DS screens to the main window. May result in better frame pacing. Mandatory when using the OpenGL 3D renderer.</p></body></html></string> +        </property> +        <property name="text"> +         <string>OpenGL display</string> +        </property> +       </widget> +      </item> +      <item row="5" column="0" colspan="2"> +       <widget class="QCheckBox" name="cbVSync"> +        <property name="whatsThis"> +         <string><html><head/><body><p>When using OpenGL, synchronize the video output to your monitor's refresh rate.</p></body></html></string> +        </property> +        <property name="text"> +         <string>VSync</string> +        </property> +       </widget> +      </item> +      <item row="3" column="0" colspan="2"> +       <spacer name="verticalSpacer"> +        <property name="orientation"> +         <enum>Qt::Vertical</enum> +        </property> +        <property name="sizeType"> +         <enum>QSizePolicy::Fixed</enum> +        </property> +        <property name="sizeHint" stdset="0"> +         <size> +          <width>20</width> +          <height>20</height> +         </size> +        </property> +       </spacer> +      </item> +      <item row="2" column="0" colspan="2"> +       <widget class="QRadioButton" name="rb3DOpenGL"> +        <property name="whatsThis"> +         <string><html><head/><body><p>The OpenGL renderer may be faster than software and supports graphical enhancements, but is more prone to glitches.</p></body></html></string> +        </property> +        <property name="text"> +         <string>OpenGL</string> +        </property> +       </widget> +      </item> +      <item row="1" column="0" colspan="2"> +       <widget class="QRadioButton" name="rb3DSoftware"> +        <property name="whatsThis"> +         <string><html><head/><body><p>The software renderer is more accurate and less prone to rendering glitches, but requires more CPU power.</p></body></html></string> +        </property> +        <property name="text"> +         <string>Software</string> +        </property> +       </widget> +      </item> +      <item row="0" column="0" colspan="2"> +       <widget class="QLabel" name="label"> +        <property name="text"> +         <string>3D renderer:</string> +        </property> +       </widget> +      </item> +     </layout> +    </widget> +   </item> +  </layout> + </widget> + <resources/> + <connections> +  <connection> +   <sender>buttonBox</sender> +   <signal>accepted()</signal> +   <receiver>VideoSettingsDialog</receiver> +   <slot>accept()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>248</x> +     <y>254</y> +    </hint> +    <hint type="destinationlabel"> +     <x>157</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> +  <connection> +   <sender>buttonBox</sender> +   <signal>rejected()</signal> +   <receiver>VideoSettingsDialog</receiver> +   <slot>reject()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>316</x> +     <y>260</y> +    </hint> +    <hint type="destinationlabel"> +     <x>286</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> + </connections> +</ui> diff --git a/src/frontend/qt_sdl/WifiSettingsDialog.cpp b/src/frontend/qt_sdl/WifiSettingsDialog.cpp new file mode 100644 index 0000000..457a78d --- /dev/null +++ b/src/frontend/qt_sdl/WifiSettingsDialog.cpp @@ -0,0 +1,140 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <stdio.h> +#include <QFileDialog> + +#include "types.h" +#include "Platform.h" +#include "Config.h" +#include "PlatformConfig.h" + +#include "LAN_Socket.h" +#include "LAN_PCap.h" +#include "Wifi.h" + +#include "WifiSettingsDialog.h" +#include "ui_WifiSettingsDialog.h" + + +#ifdef __WIN32__ +#define PCAP_NAME "winpcap/npcap" +#else +#define PCAP_NAME "libpcap" +#endif + + +WifiSettingsDialog* WifiSettingsDialog::currentDlg = nullptr; + + +WifiSettingsDialog::WifiSettingsDialog(QWidget* parent) : QDialog(parent), ui(new Ui::WifiSettingsDialog) +{ +    ui->setupUi(this); +    setAttribute(Qt::WA_DeleteOnClose); + +    LAN_Socket::Init(); +    haspcap = LAN_PCap::Init(false); + +    ui->cbDirectMode->setText("Direct mode (requires " PCAP_NAME " and ethernet connection)"); + +    ui->cbBindAnyAddr->setChecked(Config::SocketBindAnyAddr != 0); + +    int sel = 0; +    for (int i = 0; i < LAN_PCap::NumAdapters; i++) +    { +        LAN_PCap::AdapterData* adapter = &LAN_PCap::Adapters[i]; + +        ui->cbxDirectAdapter->addItem(QString(adapter->FriendlyName)); + +        if (!strncmp(adapter->DeviceName, Config::LANDevice, 128)) +            sel = i; +    } +    ui->cbxDirectAdapter->setCurrentIndex(sel); + +    ui->cbDirectMode->setChecked(Config::DirectLAN != 0); +    if (!haspcap) ui->cbDirectMode->setEnabled(false); + +    updateAdapterControls(); +} + +WifiSettingsDialog::~WifiSettingsDialog() +{ +    delete ui; +} + +void WifiSettingsDialog::on_WifiSettingsDialog_accepted() +{ +    Config::SocketBindAnyAddr = ui->cbBindAnyAddr->isChecked() ? 1:0; +    Config::DirectLAN = ui->cbDirectMode->isChecked() ? 1:0; + +    int sel = ui->cbxDirectAdapter->currentIndex(); +    if (sel < 0 || sel >= LAN_PCap::NumAdapters) sel = 0; +    if (LAN_PCap::NumAdapters < 1) +    { +        Config::LANDevice[0] = '\0'; +    } +    else +    { +        strncpy(Config::LANDevice, LAN_PCap::Adapters[sel].DeviceName, 127); +        Config::LANDevice[127] = '\0'; +    } + +    Config::Save(); + +    closeDlg(); +} + +void WifiSettingsDialog::on_WifiSettingsDialog_rejected() +{ +    closeDlg(); +} + +void WifiSettingsDialog::on_cbDirectMode_stateChanged(int state) +{ +    updateAdapterControls(); +} + +void WifiSettingsDialog::on_cbxDirectAdapter_currentIndexChanged(int sel) +{ +    if (!haspcap) return; + +    if (sel < 0 || sel >= LAN_PCap::NumAdapters) return; +    if (LAN_PCap::NumAdapters < 1) return; + +    LAN_PCap::AdapterData* adapter = &LAN_PCap::Adapters[sel]; +    char tmp[64]; + +    sprintf(tmp, "MAC: %02X:%02X:%02X:%02X:%02X:%02X", +            adapter->MAC[0], adapter->MAC[1], adapter->MAC[2], +            adapter->MAC[3], adapter->MAC[4], adapter->MAC[5]); +    ui->lblAdapterMAC->setText(QString(tmp)); + +    sprintf(tmp, "IP: %d.%d.%d.%d", +            adapter->IP_v4[0], adapter->IP_v4[1], +            adapter->IP_v4[2], adapter->IP_v4[3]); +    ui->lblAdapterIP->setText(QString(tmp)); +} + +void WifiSettingsDialog::updateAdapterControls() +{ +    bool enable = haspcap && ui->cbDirectMode->isChecked(); + +    ui->cbxDirectAdapter->setEnabled(enable); +    ui->lblAdapterMAC->setEnabled(enable); +    ui->lblAdapterIP->setEnabled(enable); +} diff --git a/src/frontend/qt_sdl/WifiSettingsDialog.h b/src/frontend/qt_sdl/WifiSettingsDialog.h new file mode 100644 index 0000000..f8aad1b --- /dev/null +++ b/src/frontend/qt_sdl/WifiSettingsDialog.h @@ -0,0 +1,68 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef WIFISETTINGSDIALOG_H +#define WIFISETTINGSDIALOG_H + +#include <QDialog> + +namespace Ui { class WifiSettingsDialog; } +class WifiSettingsDialog; + +class WifiSettingsDialog : public QDialog +{ +    Q_OBJECT + +public: +    explicit WifiSettingsDialog(QWidget* parent); +    ~WifiSettingsDialog(); + +    static WifiSettingsDialog* currentDlg; +    static WifiSettingsDialog* openDlg(QWidget* parent) +    { +        if (currentDlg) +        { +            currentDlg->activateWindow(); +            return currentDlg; +        } + +        currentDlg = new WifiSettingsDialog(parent); +        currentDlg->show(); +        return currentDlg; +    } +    static void closeDlg() +    { +        currentDlg = nullptr; +    } + +private slots: +    void on_WifiSettingsDialog_accepted(); +    void on_WifiSettingsDialog_rejected(); + +    void on_cbDirectMode_stateChanged(int state); +    void on_cbxDirectAdapter_currentIndexChanged(int sel); + +private: +    Ui::WifiSettingsDialog* ui; + +    bool haspcap; + +    void updateAdapterControls(); +}; + +#endif // WIFISETTINGSDIALOG_H diff --git a/src/frontend/qt_sdl/WifiSettingsDialog.ui b/src/frontend/qt_sdl/WifiSettingsDialog.ui new file mode 100644 index 0000000..bfee1fd --- /dev/null +++ b/src/frontend/qt_sdl/WifiSettingsDialog.ui @@ -0,0 +1,165 @@ +<?xml version="1.0" encoding="UTF-8"?> +<ui version="4.0"> + <class>WifiSettingsDialog</class> + <widget class="QDialog" name="WifiSettingsDialog"> +  <property name="geometry"> +   <rect> +    <x>0</x> +    <y>0</y> +    <width>479</width> +    <height>217</height> +   </rect> +  </property> +  <property name="sizePolicy"> +   <sizepolicy hsizetype="Fixed" vsizetype="Fixed"> +    <horstretch>0</horstretch> +    <verstretch>0</verstretch> +   </sizepolicy> +  </property> +  <property name="windowTitle"> +   <string>Wifi settings - melonDS</string> +  </property> +  <layout class="QVBoxLayout" name="verticalLayout"> +   <property name="sizeConstraint"> +    <enum>QLayout::SetFixedSize</enum> +   </property> +   <item> +    <widget class="QGroupBox" name="groupBox"> +     <property name="title"> +      <string>Local</string> +     </property> +     <layout class="QGridLayout" name="gridLayout"> +      <item row="0" column="0"> +       <widget class="QCheckBox" name="cbBindAnyAddr"> +        <property name="whatsThis"> +         <string><html><head/><body><p>Enabling this allows (theoretically) playing local multiplayer games over a local network. It may or may not help make for a better connection in general.</p></body></html></string> +        </property> +        <property name="text"> +         <string>Bind socket to any address</string> +        </property> +       </widget> +      </item> +     </layout> +    </widget> +   </item> +   <item> +    <widget class="QGroupBox" name="groupBox_2"> +     <property name="title"> +      <string>Online</string> +     </property> +     <layout class="QGridLayout" name="gridLayout_2"> +      <item row="2" column="0"> +       <widget class="QLabel" name="label_2"> +        <property name="text"> +         <string>MAC address:</string> +        </property> +       </widget> +      </item> +      <item row="0" column="0" colspan="2"> +       <widget class="QCheckBox" name="cbDirectMode"> +        <property name="whatsThis"> +         <string><html><head/><body><p>Direct mode directly routes network traffic to the host network. It is the most reliable, but requires an ethernet connection.</p><p><br/></p><p>Non-direct mode uses a layer of emulation to get around this, but is more prone to problems.</p></body></html></string> +        </property> +        <property name="text"> +         <string>Direct mode [TEXT PLACEHOLDER]</string> +        </property> +       </widget> +      </item> +      <item row="1" column="1"> +       <widget class="QComboBox" name="cbxDirectAdapter"> +        <property name="sizePolicy"> +         <sizepolicy hsizetype="Expanding" vsizetype="Fixed"> +          <horstretch>0</horstretch> +          <verstretch>0</verstretch> +         </sizepolicy> +        </property> +        <property name="minimumSize"> +         <size> +          <width>350</width> +          <height>0</height> +         </size> +        </property> +        <property name="whatsThis"> +         <string><html><head/><body><p>Selects the network adapter through which to route network traffic under direct mode.</p></body></html></string> +        </property> +       </widget> +      </item> +      <item row="1" column="0"> +       <widget class="QLabel" name="label"> +        <property name="text"> +         <string>Network adapter:</string> +        </property> +       </widget> +      </item> +      <item row="3" column="0"> +       <widget class="QLabel" name="label_3"> +        <property name="text"> +         <string>IP address:</string> +        </property> +       </widget> +      </item> +      <item row="2" column="1"> +       <widget class="QLabel" name="lblAdapterMAC"> +        <property name="text"> +         <string>[PLACEHOLDER]</string> +        </property> +       </widget> +      </item> +      <item row="3" column="1"> +       <widget class="QLabel" name="lblAdapterIP"> +        <property name="text"> +         <string>[PLACEHOLDER]</string> +        </property> +       </widget> +      </item> +     </layout> +    </widget> +   </item> +   <item> +    <widget class="QDialogButtonBox" name="buttonBox"> +     <property name="orientation"> +      <enum>Qt::Horizontal</enum> +     </property> +     <property name="standardButtons"> +      <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set> +     </property> +    </widget> +   </item> +  </layout> + </widget> + <resources/> + <connections> +  <connection> +   <sender>buttonBox</sender> +   <signal>accepted()</signal> +   <receiver>WifiSettingsDialog</receiver> +   <slot>accept()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>248</x> +     <y>254</y> +    </hint> +    <hint type="destinationlabel"> +     <x>157</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> +  <connection> +   <sender>buttonBox</sender> +   <signal>rejected()</signal> +   <receiver>WifiSettingsDialog</receiver> +   <slot>reject()</slot> +   <hints> +    <hint type="sourcelabel"> +     <x>316</x> +     <y>260</y> +    </hint> +    <hint type="destinationlabel"> +     <x>286</x> +     <y>274</y> +    </hint> +   </hints> +  </connection> + </connections> +</ui> diff --git a/src/frontend/qt_sdl/font.h b/src/frontend/qt_sdl/font.h new file mode 100644 index 0000000..f2e4f87 --- /dev/null +++ b/src/frontend/qt_sdl/font.h @@ -0,0 +1,135 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef FONT_H +#define FONT_H +unsigned short font[] = { +	12, 0, 0, 0,0x0C03, 0x0E07, 0x070E, 0x039C, 0x01F8, 0x00F0, 0x00F0, 0x01F8, 0x039C, 0x070E, 0x0E07, 0x0C03, +	12, 0, 0, 0,0x01C0, 0x00E0, 0x0060, 0x0860, 0x0C60, 0x0FE0, 0x07F0, 0x0038, 0x001C, 0x000E, 0x0007, 0x0003, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	12, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, +	2, 0, 0, 0,0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0000, 0x0000, 0x0003, 0x0003, 0x0000, 0x0000, +	9, 0, 0, 0,0x01EF, 0x01EF, 0x018C, 0x01CE, 0x00E7, 0x0063, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	10, 0, 0, 0,0x00CC, 0x00CC, 0x03FF, 0x03FF, 0x00CC, 0x00CC, 0x03FF, 0x03FF, 0x00CC, 0x00CC, 0x0000, 0x0000, +	8, 0, 0, 0,0x0018, 0x00FE, 0x00FF, 0x001B, 0x007F, 0x00FE, 0x00D8, 0x00FF, 0x007F, 0x0018, 0x0000, 0x0000, +	10, 0, 0, 0,0x0306, 0x038F, 0x01CF, 0x00E6, 0x0070, 0x0038, 0x019C, 0x03CE, 0x03C7, 0x0183, 0x0000, 0x0000, +	10, 0, 0, 0,0x007C, 0x00FE, 0x00C6, 0x00EE, 0x007C, 0x037E, 0x03E7, 0x01F3, 0x03BF, 0x031E, 0x0000, 0x0000, +	4, 0, 0, 0,0x000F, 0x000F, 0x000C, 0x000E, 0x0007, 0x0003, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	4, 0, 0, 0,0x000C, 0x000E, 0x0007, 0x0003, 0x0003, 0x0003, 0x0003, 0x0007, 0x000E, 0x000C, 0x0000, 0x0000, +	4, 0, 0, 0,0x0003, 0x0007, 0x000E, 0x000C, 0x000C, 0x000C, 0x000C, 0x000E, 0x0007, 0x0003, 0x0000, 0x0000, +	10, 0, 0, 0,0x0030, 0x0333, 0x03B7, 0x01FE, 0x00FC, 0x00FC, 0x01FE, 0x03B7, 0x0333, 0x0030, 0x0000, 0x0000, +	10, 0, 0, 0,0x0030, 0x0030, 0x0030, 0x0030, 0x03FF, 0x03FF, 0x0030, 0x0030, 0x0030, 0x0030, 0x0000, 0x0000, +	4, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x000F, 0x000F, 0x000C, 0x000E, 0x0007, 0x0003, +	10, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x03FF, 0x03FF, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	3, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0007, 0x0007, 0x0007, 0x0000, 0x0000, +	10, 0, 0, 0,0x0300, 0x0380, 0x01C0, 0x00E0, 0x0070, 0x0038, 0x001C, 0x000E, 0x0007, 0x0003, 0x0000, 0x0000, +	8, 0, 0, 0,0x007E, 0x00FF, 0x00C3, 0x00C3, 0x00C3, 0x00C3, 0x00C3, 0x00C3, 0x00FF, 0x007E, 0x0000, 0x0000, +	4, 0, 0, 0,0x0006, 0x0007, 0x0007, 0x0006, 0x0006, 0x0006, 0x0006, 0x0006, 0x000F, 0x000F, 0x0000, 0x0000, +	8, 0, 0, 0,0x007E, 0x00FF, 0x00C3, 0x00C0, 0x00FE, 0x007F, 0x0003, 0x0003, 0x00FF, 0x00FF, 0x0000, 0x0000, +	8, 0, 0, 0,0x007F, 0x00FF, 0x00C0, 0x00C0, 0x007C, 0x00FC, 0x00C0, 0x00C0, 0x00FF, 0x007F, 0x0000, 0x0000, +	8, 0, 0, 0,0x0063, 0x0063, 0x0063, 0x0063, 0x0063, 0x0063, 0x00FF, 0x00FE, 0x0060, 0x0060, 0x0000, 0x0000, +	8, 0, 0, 0,0x00FF, 0x00FF, 0x0003, 0x0003, 0x007F, 0x00FF, 0x00C0, 0x00C0, 0x00FF, 0x007F, 0x0000, 0x0000, +	8, 0, 0, 0,0x007E, 0x007F, 0x0003, 0x0003, 0x007F, 0x00FF, 0x00C3, 0x00C3, 0x00FF, 0x007E, 0x0000, 0x0000, +	8, 0, 0, 0,0x00FF, 0x00FF, 0x00C0, 0x00E0, 0x0070, 0x0038, 0x001C, 0x000C, 0x000C, 0x000C, 0x0000, 0x0000, +	8, 0, 0, 0,0x007E, 0x00FF, 0x00C3, 0x00C3, 0x007E, 0x00FF, 0x00C3, 0x00C3, 0x00FF, 0x007E, 0x0000, 0x0000, +	8, 0, 0, 0,0x007E, 0x00FF, 0x00C3, 0x00C3, 0x00FF, 0x00FE, 0x00C0, 0x00C0, 0x00FE, 0x007E, 0x0000, 0x0000, +	3, 0, 0, 0,0x0000, 0x0000, 0x0007, 0x0007, 0x0000, 0x0000, 0x0000, 0x0007, 0x0007, 0x0000, 0x0000, 0x0000, +	4, 0, 0, 0,0x0000, 0x0000, 0x000E, 0x000E, 0x0000, 0x0000, 0x000C, 0x000E, 0x0007, 0x0003, 0x0000, 0x0000, +	6, 0, 0, 0,0x0030, 0x0038, 0x001C, 0x000E, 0x0007, 0x0007, 0x000E, 0x001C, 0x0038, 0x0030, 0x0000, 0x0000, +	7, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x007F, 0x007F, 0x0000, 0x0000, 0x007F, 0x007F, 0x0000, 0x0000, 0x0000, +	6, 0, 0, 0,0x0003, 0x0007, 0x000E, 0x001C, 0x0038, 0x0038, 0x001C, 0x000E, 0x0007, 0x0003, 0x0000, 0x0000, +	8, 0, 0, 0,0x007E, 0x00FF, 0x00C3, 0x00C3, 0x00F0, 0x0078, 0x0018, 0x0000, 0x0018, 0x0018, 0x0000, 0x0000, +	10, 0, 0, 0,0x00FC, 0x01FE, 0x0387, 0x0333, 0x037B, 0x03FB, 0x01F3, 0x0007, 0x03FE, 0x03FC, 0x0000, 0x0000, +	9, 0, 0, 0,0x00FE, 0x01FF, 0x0183, 0x0183, 0x0183, 0x01FF, 0x01FF, 0x0183, 0x0183, 0x0183, 0x0000, 0x0000, +	9, 0, 0, 0,0x00FF, 0x01FF, 0x0183, 0x0183, 0x00FF, 0x01FF, 0x0183, 0x0183, 0x01FF, 0x00FF, 0x0000, 0x0000, +	8, 0, 0, 0,0x00FE, 0x00FF, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x00FF, 0x00FE, 0x0000, 0x0000, +	9, 0, 0, 0,0x007F, 0x00FF, 0x01C3, 0x0183, 0x0183, 0x0183, 0x0183, 0x01C3, 0x00FF, 0x007F, 0x0000, 0x0000, +	9, 0, 0, 0,0x01FF, 0x01FF, 0x0003, 0x0003, 0x00FF, 0x00FF, 0x0003, 0x0003, 0x01FF, 0x01FF, 0x0000, 0x0000, +	9, 0, 0, 0,0x01FF, 0x01FF, 0x0003, 0x0003, 0x00FF, 0x00FF, 0x0003, 0x0003, 0x0003, 0x0003, 0x0000, 0x0000, +	9, 0, 0, 0,0x01FE, 0x01FF, 0x0003, 0x0003, 0x01F3, 0x01F3, 0x0183, 0x0183, 0x01FF, 0x00FE, 0x0000, 0x0000, +	9, 0, 0, 0,0x0183, 0x0183, 0x0183, 0x0183, 0x01FF, 0x01FF, 0x0183, 0x0183, 0x0183, 0x0183, 0x0000, 0x0000, +	6, 0, 0, 0,0x003F, 0x003F, 0x000C, 0x000C, 0x000C, 0x000C, 0x000C, 0x000C, 0x003F, 0x003F, 0x0000, 0x0000, +	9, 0, 0, 0,0x01F0, 0x01F0, 0x00C0, 0x00C0, 0x00C0, 0x00C0, 0x00C3, 0x00C3, 0x00FF, 0x007E, 0x0000, 0x0000, +	9, 0, 0, 0,0x0183, 0x01C3, 0x00E3, 0x0073, 0x003F, 0x003F, 0x0073, 0x00E3, 0x01C3, 0x0183, 0x0000, 0x0000, +	7, 0, 0, 0,0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x007F, 0x007F, 0x0000, 0x0000, +	10, 0, 0, 0,0x0303, 0x0387, 0x03CF, 0x03FF, 0x037B, 0x0333, 0x0303, 0x0303, 0x0303, 0x0303, 0x0000, 0x0000, +	10, 0, 0, 0,0x0303, 0x0307, 0x030F, 0x031F, 0x033B, 0x0373, 0x03E3, 0x03C3, 0x0383, 0x0303, 0x0000, 0x0000, +	10, 0, 0, 0,0x01FE, 0x03FF, 0x0303, 0x0303, 0x0303, 0x0303, 0x0303, 0x0303, 0x03FF, 0x01FE, 0x0000, 0x0000, +	9, 0, 0, 0,0x00FF, 0x01FF, 0x0183, 0x0183, 0x01FF, 0x00FF, 0x0003, 0x0003, 0x0003, 0x0003, 0x0000, 0x0000, +	10, 0, 0, 0,0x01FE, 0x03FF, 0x0303, 0x0303, 0x0333, 0x0373, 0x03E3, 0x01C3, 0x03FF, 0x037E, 0x0000, 0x0000, +	9, 0, 0, 0,0x00FF, 0x01FF, 0x0183, 0x0183, 0x01FF, 0x00FF, 0x0073, 0x00E3, 0x01C3, 0x0183, 0x0000, 0x0000, +	10, 0, 0, 0,0x01FE, 0x01FF, 0x0003, 0x0003, 0x01FF, 0x03FE, 0x0300, 0x0300, 0x03FE, 0x01FE, 0x0000, 0x0000, +	10, 0, 0, 0,0x03FF, 0x03FF, 0x0030, 0x0030, 0x0030, 0x0030, 0x0030, 0x0030, 0x0030, 0x0030, 0x0000, 0x0000, +	9, 0, 0, 0,0x0183, 0x0183, 0x0183, 0x0183, 0x0183, 0x0183, 0x0183, 0x0183, 0x01FF, 0x00FE, 0x0000, 0x0000, +	10, 0, 0, 0,0x0303, 0x0303, 0x0303, 0x0303, 0x0303, 0x0387, 0x01CE, 0x00FC, 0x0078, 0x0030, 0x0000, 0x0000, +	10, 0, 0, 0,0x0303, 0x0303, 0x0303, 0x0303, 0x0333, 0x037B, 0x03FF, 0x03CF, 0x0387, 0x0303, 0x0000, 0x0000, +	10, 0, 0, 0,0x0303, 0x0387, 0x01CE, 0x00FC, 0x0078, 0x0078, 0x00FC, 0x01CE, 0x0387, 0x0303, 0x0000, 0x0000, +	10, 0, 0, 0,0x0303, 0x0387, 0x01CE, 0x00FC, 0x0078, 0x0030, 0x0030, 0x0030, 0x0030, 0x0030, 0x0000, 0x0000, +	10, 0, 0, 0,0x03FF, 0x03FF, 0x01C0, 0x00E0, 0x0070, 0x0038, 0x001C, 0x000E, 0x03FF, 0x03FF, 0x0000, 0x0000, +	4, 0, 0, 0,0x000F, 0x000F, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x000F, 0x000F, 0x0000, 0x0000, +	10, 0, 0, 0,0x0003, 0x0007, 0x000E, 0x001C, 0x0038, 0x0070, 0x00E0, 0x01C0, 0x0380, 0x0300, 0x0000, 0x0000, +	4, 0, 0, 0,0x000F, 0x000F, 0x000C, 0x000C, 0x000C, 0x000C, 0x000C, 0x000C, 0x000F, 0x000F, 0x0000, 0x0000, +	8, 0, 0, 0,0x0018, 0x003C, 0x007E, 0x00E7, 0x00C3, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	10, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x03FF, 0x03FF, +	4, 0, 0, 0,0x000F, 0x000F, 0x0003, 0x0007, 0x000E, 0x000C, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	7, 0, 0, 0,0x0000, 0x0000, 0x003E, 0x007E, 0x0060, 0x007E, 0x007F, 0x0063, 0x007F, 0x007E, 0x0000, 0x0000, +	7, 0, 0, 0,0x0003, 0x0003, 0x0003, 0x003F, 0x007F, 0x0063, 0x0063, 0x0063, 0x007F, 0x003F, 0x0000, 0x0000, +	7, 0, 0, 0,0x0000, 0x0000, 0x007E, 0x007F, 0x0003, 0x0003, 0x0003, 0x0003, 0x007F, 0x007E, 0x0000, 0x0000, +	7, 0, 0, 0,0x0060, 0x0060, 0x0060, 0x007E, 0x007F, 0x0063, 0x0063, 0x0063, 0x007F, 0x007E, 0x0000, 0x0000, +	7, 0, 0, 0,0x0000, 0x0000, 0x003E, 0x007F, 0x0063, 0x007F, 0x003F, 0x0003, 0x003F, 0x003E, 0x0000, 0x0000, +	6, 0, 0, 0,0x003C, 0x003E, 0x0006, 0x0006, 0x001F, 0x001F, 0x0006, 0x0006, 0x0006, 0x0006, 0x0000, 0x0000, +	7, 0, 0, 0,0x0000, 0x0000, 0x007E, 0x007F, 0x0063, 0x0063, 0x007F, 0x007E, 0x0060, 0x0060, 0x007E, 0x003E, +	7, 0, 0, 0,0x0003, 0x0003, 0x0003, 0x003F, 0x007F, 0x0063, 0x0063, 0x0063, 0x0063, 0x0063, 0x0000, 0x0000, +	2, 0, 0, 0,0x0003, 0x0003, 0x0000, 0x0000, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0000, 0x0000, +	7, 0, 0, 0,0x0060, 0x0060, 0x0000, 0x0000, 0x0060, 0x0060, 0x0060, 0x0060, 0x0060, 0x0063, 0x007F, 0x003E, +	8, 0, 0, 0,0x0003, 0x0003, 0x00E3, 0x0073, 0x003B, 0x001F, 0x001F, 0x003B, 0x0073, 0x00E3, 0x0000, 0x0000, +	4, 0, 0, 0,0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x000F, 0x000E, 0x0000, 0x0000, +	10, 0, 0, 0,0x0000, 0x0000, 0x01FF, 0x03FF, 0x0333, 0x0333, 0x0333, 0x0333, 0x0333, 0x0333, 0x0000, 0x0000, +	7, 0, 0, 0,0x0000, 0x0000, 0x003F, 0x007F, 0x0063, 0x0063, 0x0063, 0x0063, 0x0063, 0x0063, 0x0000, 0x0000, +	8, 0, 0, 0,0x0000, 0x0000, 0x007E, 0x00FF, 0x00C3, 0x00C3, 0x00C3, 0x00C3, 0x00FF, 0x007E, 0x0000, 0x0000, +	7, 0, 0, 0,0x0000, 0x0000, 0x003F, 0x007F, 0x0063, 0x0063, 0x007F, 0x003F, 0x0003, 0x0003, 0x0003, 0x0003, +	7, 0, 0, 0,0x0000, 0x0000, 0x007E, 0x007F, 0x0063, 0x0063, 0x007F, 0x007E, 0x0060, 0x0060, 0x0060, 0x0060, +	7, 0, 0, 0,0x0000, 0x0000, 0x003B, 0x007F, 0x0067, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0000, 0x0000, +	8, 0, 0, 0,0x0000, 0x0000, 0x007E, 0x007F, 0x0003, 0x007F, 0x00FE, 0x00C0, 0x00FE, 0x007E, 0x0000, 0x0000, +	6, 0, 0, 0,0x0006, 0x0006, 0x003F, 0x003F, 0x0006, 0x0006, 0x0006, 0x0006, 0x003E, 0x003C, 0x0000, 0x0000, +	7, 0, 0, 0,0x0000, 0x0000, 0x0063, 0x0063, 0x0063, 0x0063, 0x0063, 0x0063, 0x007F, 0x007E, 0x0000, 0x0000, +	10, 0, 0, 0,0x0000, 0x0000, 0x0303, 0x0303, 0x0303, 0x0387, 0x01CE, 0x00FC, 0x0078, 0x0030, 0x0000, 0x0000, +	10, 0, 0, 0,0x0000, 0x0000, 0x0303, 0x0303, 0x0333, 0x037B, 0x03FF, 0x03CF, 0x0387, 0x0303, 0x0000, 0x0000, +	8, 0, 0, 0,0x0000, 0x0000, 0x00C3, 0x00E7, 0x007E, 0x003C, 0x003C, 0x007E, 0x00E7, 0x00C3, 0x0000, 0x0000, +	10, 0, 0, 0,0x0000, 0x0000, 0x0303, 0x0307, 0x038E, 0x01DC, 0x00F8, 0x0070, 0x0038, 0x001C, 0x000E, 0x0006, +	8, 0, 0, 0,0x0000, 0x0000, 0x00FF, 0x00FF, 0x0070, 0x0038, 0x001C, 0x000E, 0x00FF, 0x00FF, 0x0000, 0x0000, +	6, 0, 0, 0,0x0038, 0x003C, 0x000C, 0x000C, 0x000F, 0x000F, 0x000C, 0x000C, 0x003C, 0x0038, 0x0000, 0x0000, +	2, 0, 0, 0,0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0003, 0x0000, 0x0000, +	6, 0, 0, 0,0x0007, 0x000F, 0x000C, 0x000C, 0x003C, 0x003C, 0x000C, 0x000C, 0x000F, 0x0007, 0x0000, 0x0000, +	10, 0, 0, 0,0x0000, 0x0000, 0x0000, 0x031C, 0x03BE, 0x01F7, 0x00E3, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, +	11, 0, 0, 0,0x0555, 0x0000, 0x0401, 0x0000, 0x0401, 0x0000, 0x0401, 0x0000, 0x0401, 0x0000, 0x0555, 0x0000, +}; +#endif diff --git a/src/frontend/qt_sdl/main.cpp b/src/frontend/qt_sdl/main.cpp new file mode 100644 index 0000000..68a32c8 --- /dev/null +++ b/src/frontend/qt_sdl/main.cpp @@ -0,0 +1,2100 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#include <stdlib.h> +#include <time.h> +#include <stdio.h> +#include <string.h> + +#include <QApplication> +#include <QMessageBox> +#include <QMenuBar> +#include <QFileDialog> +#include <QPaintEvent> +#include <QPainter> +#include <QKeyEvent> +#include <QMimeData> + +#include <SDL2/SDL.h> + +#include "main.h" +#include "Input.h" +#include "EmuSettingsDialog.h" +#include "InputConfigDialog.h" +#include "VideoSettingsDialog.h" +#include "AudioSettingsDialog.h" +#include "WifiSettingsDialog.h" + +#include "types.h" +#include "version.h" + +#include "FrontendUtil.h" +#include "OSD.h" + +#include "NDS.h" +#include "GBACart.h" +#include "OpenGLSupport.h" +#include "GPU.h" +#include "SPU.h" +#include "Wifi.h" +#include "Platform.h" +#include "Config.h" +#include "PlatformConfig.h" + +#include "Savestate.h" + +#include "main_shaders.h" + + +// TODO: uniform variable spelling + +bool RunningSomething; + +MainWindow* mainWindow; +EmuThread* emuThread; + +int autoScreenSizing = 0; + +int videoRenderer; +GPU::RenderSettings videoSettings; +bool videoSettingsDirty; + +SDL_AudioDeviceID audioDevice; +int audioFreq; +SDL_cond* audioSync; +SDL_mutex* audioSyncLock; + +SDL_AudioDeviceID micDevice; +s16 micExtBuffer[2048]; +u32 micExtBufferWritePos; + +u32 micWavLength; +s16* micWavBuffer; + + +void audioCallback(void* data, Uint8* stream, int len) +{ +    len /= (sizeof(s16) * 2); + +    // resample incoming audio to match the output sample rate + +    int len_in = Frontend::AudioOut_GetNumSamples(len); +    s16 buf_in[1024*2]; +    int num_in; + +    SDL_LockMutex(audioSyncLock); +    num_in = SPU::ReadOutput(buf_in, len_in); +    SDL_CondSignal(audioSync); +    SDL_UnlockMutex(audioSyncLock); + +    if (num_in < 1) +    { +        memset(stream, 0, len*sizeof(s16)*2); +        return; +    } + +    int margin = 6; +    if (num_in < len_in-margin) +    { +        int last = num_in-1; +        if (last < 0) last = 0; + +        for (int i = num_in; i < len_in-margin; i++) +            ((u32*)buf_in)[i] = ((u32*)buf_in)[last]; + +        num_in = len_in-margin; +    } + +    Frontend::AudioOut_Resample(buf_in, num_in, (s16*)stream, len, Config::AudioVolume); +} + + +void micLoadWav(const char* name) +{ +    SDL_AudioSpec format; +    memset(&format, 0, sizeof(SDL_AudioSpec)); + +    if (micWavBuffer) delete[] micWavBuffer; +    micWavBuffer = nullptr; +    micWavLength = 0; + +    u8* buf; +    u32 len; +    if (!SDL_LoadWAV(name, &format, &buf, &len)) +        return; + +    const u64 dstfreq = 44100; + +    if (format.format == AUDIO_S16 || format.format == AUDIO_U16) +    { +        int srcinc = format.channels; +        len /= (2 * srcinc); + +        micWavLength = (len * dstfreq) / format.freq; +        if (micWavLength < 735) micWavLength = 735; +        micWavBuffer = new s16[micWavLength]; + +        float res_incr = len / (float)micWavLength; +        float res_timer = 0; +        int res_pos = 0; + +        for (int i = 0; i < micWavLength; i++) +        { +            u16 val = ((u16*)buf)[res_pos]; +            if (SDL_AUDIO_ISUNSIGNED(format.format)) val ^= 0x8000; + +            micWavBuffer[i] = val; + +            res_timer += res_incr; +            while (res_timer >= 1.0) +            { +                res_timer -= 1.0; +                res_pos += srcinc; +            } +        } +    } +    else if (format.format == AUDIO_S8 || format.format == AUDIO_U8) +    { +        int srcinc = format.channels; +        len /= srcinc; + +        micWavLength = (len * dstfreq) / format.freq; +        if (micWavLength < 735) micWavLength = 735; +        micWavBuffer = new s16[micWavLength]; + +        float res_incr = len / (float)micWavLength; +        float res_timer = 0; +        int res_pos = 0; + +        for (int i = 0; i < micWavLength; i++) +        { +            u16 val = buf[res_pos] << 8; +            if (SDL_AUDIO_ISUNSIGNED(format.format)) val ^= 0x8000; + +            micWavBuffer[i] = val; + +            res_timer += res_incr; +            while (res_timer >= 1.0) +            { +                res_timer -= 1.0; +                res_pos += srcinc; +            } +        } +    } +    else +        printf("bad WAV format %08X\n", format.format); + +    SDL_FreeWAV(buf); +} + +void micCallback(void* data, Uint8* stream, int len) +{ +    s16* input = (s16*)stream; +    len /= sizeof(s16); + +    int maxlen = sizeof(micExtBuffer) / sizeof(s16); + +    if ((micExtBufferWritePos + len) > maxlen) +    { +        u32 len1 = maxlen - micExtBufferWritePos; +        memcpy(&micExtBuffer[micExtBufferWritePos], &input[0], len1*sizeof(s16)); +        memcpy(&micExtBuffer[0], &input[len1], (len - len1)*sizeof(s16)); +        micExtBufferWritePos = len - len1; +    } +    else +    { +        memcpy(&micExtBuffer[micExtBufferWritePos], input, len*sizeof(s16)); +        micExtBufferWritePos += len; +    } +} + +void micProcess() +{ +    int type = Config::MicInputType; +    bool cmd = Input::HotkeyDown(HK_Mic); + +    if (type != 1 && !cmd) +    { +        type = 0; +    } + +    switch (type) +    { +    case 0: // no mic +        Frontend::Mic_FeedSilence(); +        break; + +    case 1: // host mic +    case 3: // WAV +        Frontend::Mic_FeedExternalBuffer(); +        break; + +    case 2: // white noise +        Frontend::Mic_FeedNoise(); +        break; +    } +} + + +EmuThread::EmuThread(QObject* parent) : QThread(parent) +{ +    EmuStatus = 0; +    EmuRunning = 2; +    RunningSomething = false; + +    connect(this, SIGNAL(windowUpdate()), mainWindow->panel, SLOT(update())); +    connect(this, SIGNAL(windowTitleChange(QString)), mainWindow, SLOT(onTitleUpdate(QString))); +    connect(this, SIGNAL(windowEmuStart()), mainWindow, SLOT(onEmuStart())); +    connect(this, SIGNAL(windowEmuStop()), mainWindow, SLOT(onEmuStop())); +    connect(this, SIGNAL(windowEmuPause()), mainWindow->actPause, SLOT(trigger())); +    connect(this, SIGNAL(windowEmuReset()), mainWindow->actReset, SLOT(trigger())); +    connect(this, SIGNAL(screenLayoutChange()), mainWindow->panel, SLOT(onScreenLayoutChanged())); + +    if (mainWindow->hasOGL) initOpenGL(); +} + +void EmuThread::initOpenGL() +{ +    QOpenGLContext* windowctx = mainWindow->getOGLContext(); +    QSurfaceFormat format = windowctx->format(); + +    format.setSwapInterval(0); + +    oglSurface = new QOffscreenSurface(); +    oglSurface->setFormat(format); +    oglSurface->create(); +    if (!oglSurface->isValid()) +    { +        // TODO handle this! +        printf("oglSurface shat itself :(\n"); +        delete oglSurface; +        return; +    } + +    oglContext = new QOpenGLContext(); +    oglContext->setFormat(oglSurface->format()); +    oglContext->setShareContext(windowctx); +    if (!oglContext->create()) +    { +        // TODO handle this! +        printf("oglContext shat itself :(\n"); +        delete oglContext; +        delete oglSurface; +        return; +    } + +    oglContext->moveToThread(this); +} + +void EmuThread::deinitOpenGL() +{ +    delete oglContext; +    delete oglSurface; +} + +void* oglGetProcAddress(const char* proc) +{ +    return emuThread->oglGetProcAddress(proc); +} + +void* EmuThread::oglGetProcAddress(const char* proc) +{ +    return (void*)oglContext->getProcAddress(proc); +} + +void EmuThread::run() +{ +    bool hasOGL = mainWindow->hasOGL; +    u32 mainScreenPos[3]; + +    NDS::Init(); + +    mainScreenPos[0] = 0; +    mainScreenPos[1] = 0; +    mainScreenPos[2] = 0; +    autoScreenSizing = 0; + +    videoSettingsDirty = false; +    videoSettings.Soft_Threaded = Config::Threaded3D != 0; +    videoSettings.GL_ScaleFactor = Config::GL_ScaleFactor; + +    if (hasOGL) +    { +        oglContext->makeCurrent(oglSurface); +        videoRenderer = OpenGL::Init() ? Config::_3DRenderer : 0; +    } +    else +        videoRenderer = 0; + +    GPU::InitRenderer(videoRenderer); +    GPU::SetRenderSettings(videoRenderer, videoSettings); + +    Input::Init(); + +    u32 nframes = 0; +    u32 starttick = SDL_GetTicks(); +    u32 lasttick = starttick; +    u32 lastmeasuretick = lasttick; +    u32 fpslimitcount = 0; + +    char melontitle[100]; + +    while (EmuRunning != 0) +    { +        Input::Process(); + +        if (Input::HotkeyPressed(HK_FastForwardToggle)) emit windowLimitFPSChange(); + +        if (Input::HotkeyPressed(HK_Pause)) emit windowEmuPause(); +        if (Input::HotkeyPressed(HK_Reset)) emit windowEmuReset(); + +        if (GBACart::CartInserted && GBACart::HasSolarSensor) +        { +            if (Input::HotkeyPressed(HK_SolarSensorDecrease)) +            { +                if (GBACart_SolarSensor::LightLevel > 0) GBACart_SolarSensor::LightLevel--; +                char msg[64]; +                sprintf(msg, "Solar sensor level set to %d", GBACart_SolarSensor::LightLevel); +                OSD::AddMessage(0, msg); +            } +            if (Input::HotkeyPressed(HK_SolarSensorIncrease)) +            { +                if (GBACart_SolarSensor::LightLevel < 10) GBACart_SolarSensor::LightLevel++; +                char msg[64]; +                sprintf(msg, "Solar sensor level set to %d", GBACart_SolarSensor::LightLevel); +                OSD::AddMessage(0, msg); +            } +        } + +        if (EmuRunning == 1) +        { +            EmuStatus = 1; + +            // update render settings if needed +            if (videoSettingsDirty) +            { +                if (hasOGL != mainWindow->hasOGL) +                { +                    hasOGL = mainWindow->hasOGL; +                    if (hasOGL) +                    { +                        oglContext->makeCurrent(oglSurface); +                        videoRenderer = OpenGL::Init() ? Config::_3DRenderer : 0; +                    } +                    else +                        videoRenderer = 0; +                } +                else +                    videoRenderer = hasOGL ? Config::_3DRenderer : 0; + +                videoSettingsDirty = false; +                videoSettings.Soft_Threaded = Config::Threaded3D != 0; +                videoSettings.GL_ScaleFactor = Config::GL_ScaleFactor; +                GPU::SetRenderSettings(videoRenderer, videoSettings); +            } + +            // process input and hotkeys +            NDS::SetKeyMask(Input::InputMask); + +            if (Input::HotkeyPressed(HK_Lid)) +            { +                bool lid = !NDS::IsLidClosed(); +                NDS::SetLidClosed(lid); +                OSD::AddMessage(0, lid ? "Lid closed" : "Lid opened"); +            } + +            // microphone input +            micProcess(); + +            // auto screen layout +            if (Config::ScreenSizing == 3) +            { +                mainScreenPos[2] = mainScreenPos[1]; +                mainScreenPos[1] = mainScreenPos[0]; +                mainScreenPos[0] = NDS::PowerControl9 >> 15; + +                int guess; +                if (mainScreenPos[0] == mainScreenPos[2] && +                    mainScreenPos[0] != mainScreenPos[1]) +                { +                    // constant flickering, likely displaying 3D on both screens +                    // TODO: when both screens are used for 2D only...??? +                    guess = 0; +                } +                else +                { +                    if (mainScreenPos[0] == 1) +                        guess = 1; +                    else +                        guess = 2; +                } + +                if (guess != autoScreenSizing) +                { +                    autoScreenSizing = guess; +                    emit screenLayoutChange(); +                } +            } + +            // emulate +            u32 nlines = NDS::RunFrame(); + +#ifdef MELONCAP +            MelonCap::Update(); +#endif // MELONCAP + +            if (EmuRunning == 0) break; + +            emit windowUpdate(); + +            bool fastforward = Input::HotkeyDown(HK_FastForward); + +            if (Config::AudioSync && (!fastforward) && audioDevice) +            { +                SDL_LockMutex(audioSyncLock); +                while (SPU::GetOutputSize() > 1024) +                { +                    int ret = SDL_CondWaitTimeout(audioSync, audioSyncLock, 500); +                    if (ret == SDL_MUTEX_TIMEDOUT) break; +                } +                SDL_UnlockMutex(audioSyncLock); +            } + +            float framerate = (1000.0f * nlines) / (60.0f * 263.0f); + +            { +                u32 curtick = SDL_GetTicks(); +                u32 delay = curtick - lasttick; + +                bool limitfps = Config::LimitFPS && !fastforward; +                if (limitfps) +                { +                    float wantedtickF = starttick + (framerate * (fpslimitcount+1)); +                    u32 wantedtick = (u32)ceil(wantedtickF); +                    if (curtick < wantedtick) SDL_Delay(wantedtick - curtick); + +                    lasttick = SDL_GetTicks(); +                    fpslimitcount++; +                    if ((abs(wantedtickF - (float)wantedtick) < 0.001312) || (fpslimitcount > 60)) +                    { +                        fpslimitcount = 0; +                        starttick = lasttick; +                    } +                } +                else +                { +                    if (delay < 1) SDL_Delay(1); +                    lasttick = SDL_GetTicks(); +                } +            } + +            nframes++; +            if (nframes >= 30) +            { +                u32 tick = SDL_GetTicks(); +                u32 diff = tick - lastmeasuretick; +                lastmeasuretick = tick; + +                u32 fps; +                if (diff < 1) fps = 77777; +                else fps = (nframes * 1000) / diff; +                nframes = 0; + +                float fpstarget; +                if (framerate < 1) fpstarget = 999; +                else fpstarget = 1000.0f/framerate; + +                sprintf(melontitle, "[%d/%.0f] melonDS " MELONDS_VERSION, fps, fpstarget); +                changeWindowTitle(melontitle); +            } +        } +        else +        { +            // paused +            nframes = 0; +            lasttick = SDL_GetTicks(); +            starttick = lasttick; +            lastmeasuretick = lasttick; +            fpslimitcount = 0; + +            emit windowUpdate(); + +            EmuStatus = EmuRunning; + +            sprintf(melontitle, "melonDS " MELONDS_VERSION); +            changeWindowTitle(melontitle); + +            SDL_Delay(75); +        } +    } + +    EmuStatus = 0; + +    GPU::DeInitRenderer(); +    NDS::DeInit(); +    //Platform::LAN_DeInit(); + +    if (hasOGL) +    { +        oglContext->doneCurrent(); +        deinitOpenGL(); +    } +} + +void EmuThread::changeWindowTitle(char* title) +{ +    emit windowTitleChange(QString(title)); +} + +void EmuThread::emuRun() +{ +    EmuRunning = 1; +    RunningSomething = true; + +    // checkme +    emit windowEmuStart(); +    if (audioDevice) SDL_PauseAudioDevice(audioDevice, 0); +    if (micDevice)   SDL_PauseAudioDevice(micDevice, 0); +} + +void EmuThread::emuPause() +{ +    PrevEmuStatus = EmuRunning; +    EmuRunning = 2; +    while (EmuStatus != 2); + +    if (audioDevice) SDL_PauseAudioDevice(audioDevice, 1); +    if (micDevice)   SDL_PauseAudioDevice(micDevice, 1); +} + +void EmuThread::emuUnpause() +{ +    EmuRunning = PrevEmuStatus; + +    if (audioDevice) SDL_PauseAudioDevice(audioDevice, 0); +    if (micDevice)   SDL_PauseAudioDevice(micDevice, 0); +} + +void EmuThread::emuStop() +{ +    EmuRunning = 0; + +    if (audioDevice) SDL_PauseAudioDevice(audioDevice, 1); +    if (micDevice)   SDL_PauseAudioDevice(micDevice, 1); +} + +bool EmuThread::emuIsRunning() +{ +    return (EmuRunning == 1); +} + + +void ScreenHandler::screenSetupLayout(int w, int h) +{ +    int sizing = Config::ScreenSizing; +    if (sizing == 3) sizing = autoScreenSizing; + +    Frontend::SetupScreenLayout(w, h, +                                Config::ScreenLayout, +                                Config::ScreenRotation, +                                sizing, +                                Config::ScreenGap, +                                Config::IntegerScaling != 0); + +    Frontend::GetScreenTransforms(screenMatrix[0], screenMatrix[1]); +} + +QSize ScreenHandler::screenGetMinSize() +{ +    bool isHori = (Config::ScreenRotation == 1 || Config::ScreenRotation == 3); +    int gap = Config::ScreenGap; + +    int w = 256; +    int h = 192; + +    if (Config::ScreenLayout == 0) // natural +    { +        if (isHori) +            return QSize(h+gap+h, w); +        else +            return QSize(w, h+gap+h); +    } +    else if (Config::ScreenLayout == 1) // vertical +    { +        if (isHori) +            return QSize(h, w+gap+w); +        else +            return QSize(w, h+gap+h); +    } +    else // horizontal +    { +        if (isHori) +            return QSize(h+gap+h, w); +        else +            return QSize(w+gap+w, h); +    } +} + +void ScreenHandler::screenOnMousePress(QMouseEvent* event) +{ +    event->accept(); +    if (event->button() != Qt::LeftButton) return; + +    int x = event->pos().x(); +    int y = event->pos().y(); + +    Frontend::GetTouchCoords(x, y); + +    if (x >= 0 && x < 256 && y >= 0 && y < 192) +    { +        touching = true; +        NDS::TouchScreen(x, y); +    } +} + +void ScreenHandler::screenOnMouseRelease(QMouseEvent* event) +{ +    event->accept(); +    if (event->button() != Qt::LeftButton) return; + +    if (touching) +    { +        touching = false; +        NDS::ReleaseScreen(); +    } +} + +void ScreenHandler::screenOnMouseMove(QMouseEvent* event) +{ +    event->accept(); +    if (!(event->buttons() & Qt::LeftButton)) return; +    if (!touching) return; + +    int x = event->pos().x(); +    int y = event->pos().y(); + +    Frontend::GetTouchCoords(x, y); + +    // clamp to screen range +    if (x < 0) x = 0; +    else if (x > 255) x = 255; +    if (y < 0) y = 0; +    else if (y > 191) y = 191; + +    NDS::TouchScreen(x, y); +} + + +ScreenPanelNative::ScreenPanelNative(QWidget* parent) : QWidget(parent) +{ +    screen[0] = QImage(256, 192, QImage::Format_RGB32); +    screen[1] = QImage(256, 192, QImage::Format_RGB32); + +    screenTrans[0].reset(); +    screenTrans[1].reset(); + +    touching = false; + +    OSD::Init(nullptr); +} + +ScreenPanelNative::~ScreenPanelNative() +{ +    OSD::DeInit(nullptr); +} + +void ScreenPanelNative::setupScreenLayout() +{ +    int w = width(); +    int h = height(); +    float* mtx; + +    screenSetupLayout(w, h); + +    mtx = screenMatrix[0]; +    screenTrans[0].setMatrix(mtx[0], mtx[1], 0.f, +                             mtx[2], mtx[3], 0.f, +                             mtx[4], mtx[5], 1.f); + +    mtx = screenMatrix[1]; +    screenTrans[1].setMatrix(mtx[0], mtx[1], 0.f, +                             mtx[2], mtx[3], 0.f, +                             mtx[4], mtx[5], 1.f); +} + +void ScreenPanelNative::paintEvent(QPaintEvent* event) +{ +    QPainter painter(this); + +    // fill background +    painter.fillRect(event->rect(), QColor::fromRgb(0, 0, 0)); + +    int frontbuf = GPU::FrontBuffer; +    if (!GPU::Framebuffer[frontbuf][0] || !GPU::Framebuffer[frontbuf][1]) return; + +    memcpy(screen[0].scanLine(0), GPU::Framebuffer[frontbuf][0], 256*192*4); +    memcpy(screen[1].scanLine(0), GPU::Framebuffer[frontbuf][1], 256*192*4); + +    painter.setRenderHint(QPainter::SmoothPixmapTransform, Config::ScreenFilter!=0); + +    QRect screenrc(0, 0, 256, 192); + +    painter.setTransform(screenTrans[0]); +    painter.drawImage(screenrc, screen[0]); + +    painter.setTransform(screenTrans[1]); +    painter.drawImage(screenrc, screen[1]); + +    OSD::Update(nullptr); +    OSD::DrawNative(painter); +} + +void ScreenPanelNative::resizeEvent(QResizeEvent* event) +{ +    setupScreenLayout(); +} + +void ScreenPanelNative::mousePressEvent(QMouseEvent* event) +{ +    screenOnMousePress(event); +} + +void ScreenPanelNative::mouseReleaseEvent(QMouseEvent* event) +{ +    screenOnMouseRelease(event); +} + +void ScreenPanelNative::mouseMoveEvent(QMouseEvent* event) +{ +    screenOnMouseMove(event); +} + +void ScreenPanelNative::onScreenLayoutChanged() +{ +    setMinimumSize(screenGetMinSize()); +    setupScreenLayout(); +} + + +ScreenPanelGL::ScreenPanelGL(QWidget* parent) : QOpenGLWidget(parent) +{ +    touching = false; + +} + +ScreenPanelGL::~ScreenPanelGL() +{ +    makeCurrent(); + +    OSD::DeInit(this); + +    glDeleteTextures(1, &screenTexture); + +    glDeleteVertexArrays(1, &screenVertexArray); +    glDeleteBuffers(1, &screenVertexBuffer); + +    delete screenShader; + +    doneCurrent(); +} + +void ScreenPanelGL::setupScreenLayout() +{ +    int w = width(); +    int h = height(); + +    screenSetupLayout(w, h); +} + +void ScreenPanelGL::initializeGL() +{ +    initializeOpenGLFunctions(); + +    const GLubyte* renderer = glGetString(GL_RENDERER); // get renderer string +    const GLubyte* version = glGetString(GL_VERSION); // version as a string +    printf("OpenGL: renderer: %s\n", renderer); +    printf("OpenGL: version: %s\n", version); + +    glClearColor(0, 0, 0, 1); + +    screenShader = new QOpenGLShaderProgram(this); +    screenShader->addShaderFromSourceCode(QOpenGLShader::Vertex, kScreenVS); +    screenShader->addShaderFromSourceCode(QOpenGLShader::Fragment, kScreenFS); + +    GLuint pid = screenShader->programId(); +    glBindAttribLocation(pid, 0, "vPosition"); +    glBindAttribLocation(pid, 1, "vTexcoord"); +    glBindFragDataLocation(pid, 0, "oColor"); + +    screenShader->link(); + +    screenShader->bind(); +    screenShader->setUniformValue("ScreenTex", (GLint)0); +    screenShader->release(); + + +    float vertices[] = +    { +        0,   0,    0, 0, +        0,   192,  0, 0.5, +        256, 192,  1, 0.5, +        0,   0,    0, 0, +        256, 192,  1, 0.5, +        256, 0,    1, 0, + +        0,   0,    0, 0.5, +        0,   192,  0, 1, +        256, 192,  1, 1, +        0,   0,    0, 0.5, +        256, 192,  1, 1, +        256, 0,    1, 0.5 +    }; + +    glGenBuffers(1, &screenVertexBuffer); +    glBindBuffer(GL_ARRAY_BUFFER, screenVertexBuffer); +    glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW); + +    glGenVertexArrays(1, &screenVertexArray); +    glBindVertexArray(screenVertexArray); +    glEnableVertexAttribArray(0); // position +    glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 4*4, (void*)(0)); +    glEnableVertexAttribArray(1); // texcoord +    glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, 4*4, (void*)(2*4)); + +    glGenTextures(1, &screenTexture); +    glActiveTexture(GL_TEXTURE0); +    glBindTexture(GL_TEXTURE_2D, screenTexture); +    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); +    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); +    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); +    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); +    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, 256, 192*2, 0, GL_RGBA, GL_UNSIGNED_BYTE, NULL); + +    OSD::Init(this); +} + +void ScreenPanelGL::paintGL() +{ +    int w = width(); +    int h = height(); +    float factor = devicePixelRatioF(); + +    glClear(GL_COLOR_BUFFER_BIT); + +    glViewport(0, 0, w*factor, h*factor); + +    screenShader->bind(); + +    screenShader->setUniformValue("uScreenSize", (float)w*factor, (float)h*factor); + +    int frontbuf = GPU::FrontBuffer; +    glActiveTexture(GL_TEXTURE0); + +    if (GPU::Renderer != 0) +    { +        // hardware-accelerated render +        GPU::GLCompositor::BindOutputTexture(); +    } +    else +    { +        // regular render +        glBindTexture(GL_TEXTURE_2D, screenTexture); + +        if (GPU::Framebuffer[frontbuf][0] && GPU::Framebuffer[frontbuf][1]) +        { +            glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, 256, 192, GL_RGBA, +                            GL_UNSIGNED_BYTE, GPU::Framebuffer[frontbuf][0]); +            glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 192, 256, 192, GL_RGBA, +                            GL_UNSIGNED_BYTE, GPU::Framebuffer[frontbuf][1]); +        } +    } + +    GLint filter = Config::ScreenFilter ? GL_LINEAR : GL_NEAREST; +    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, filter); +    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, filter); + +    glBindBuffer(GL_ARRAY_BUFFER, screenVertexBuffer); +    glBindVertexArray(screenVertexArray); + +    GLint transloc = screenShader->uniformLocation("uTransform"); + +    glUniformMatrix2x3fv(transloc, 1, GL_TRUE, screenMatrix[0]); +    glDrawArrays(GL_TRIANGLES, 0, 2*3); + +    glUniformMatrix2x3fv(transloc, 1, GL_TRUE, screenMatrix[1]); +    glDrawArrays(GL_TRIANGLES, 2*3, 2*3); + +    screenShader->release(); + +    OSD::Update(this); +    OSD::DrawGL(this, w*factor, h*factor); +} + +void ScreenPanelGL::resizeEvent(QResizeEvent* event) +{ +    setupScreenLayout(); + +    QOpenGLWidget::resizeEvent(event); +} + +void ScreenPanelGL::resizeGL(int w, int h) +{ +} + +void ScreenPanelGL::mousePressEvent(QMouseEvent* event) +{ +    screenOnMousePress(event); +} + +void ScreenPanelGL::mouseReleaseEvent(QMouseEvent* event) +{ +    screenOnMouseRelease(event); +} + +void ScreenPanelGL::mouseMoveEvent(QMouseEvent* event) +{ +    screenOnMouseMove(event); +} + +void ScreenPanelGL::onScreenLayoutChanged() +{ +    setMinimumSize(screenGetMinSize()); +    setupScreenLayout(); +} + + +MainWindow::MainWindow(QWidget* parent) : QMainWindow(parent) +{ +    setWindowTitle("melonDS " MELONDS_VERSION); +    setAttribute(Qt::WA_DeleteOnClose); +    setAcceptDrops(true); + +    QMenuBar* menubar = new QMenuBar(); +    { +        QMenu* menu = menubar->addMenu("File"); + +        actOpenROM = menu->addAction("Open ROM..."); +        connect(actOpenROM, &QAction::triggered, this, &MainWindow::onOpenFile); + +        //actBootFirmware = menu->addAction("Launch DS menu"); +        actBootFirmware = menu->addAction("Boot firmware"); +        connect(actBootFirmware, &QAction::triggered, this, &MainWindow::onBootFirmware); + +        menu->addSeparator(); + +        { +            QMenu* submenu = menu->addMenu("Save state"); + +            for (int i = 1; i < 9; i++) +            { +                actSaveState[i] = submenu->addAction(QString("%1").arg(i)); +                actSaveState[i]->setShortcut(QKeySequence(Qt::ShiftModifier | (Qt::Key_F1+i-1))); +                actSaveState[i]->setData(QVariant(i)); +                connect(actSaveState[i], &QAction::triggered, this, &MainWindow::onSaveState); +            } + +            actSaveState[0] = submenu->addAction("File..."); +            actSaveState[0]->setShortcut(QKeySequence(Qt::ShiftModifier | Qt::Key_F9)); +            actSaveState[0]->setData(QVariant(0)); +            connect(actSaveState[0], &QAction::triggered, this, &MainWindow::onSaveState); +        } +        { +            QMenu* submenu = menu->addMenu("Load state"); + +            for (int i = 1; i < 9; i++) +            { +                actLoadState[i] = submenu->addAction(QString("%1").arg(i)); +                actLoadState[i]->setShortcut(QKeySequence(Qt::Key_F1+i-1)); +                actLoadState[i]->setData(QVariant(i)); +                connect(actLoadState[i], &QAction::triggered, this, &MainWindow::onLoadState); +            } + +            actLoadState[0] = submenu->addAction("File..."); +            actLoadState[0]->setShortcut(QKeySequence(Qt::Key_F9)); +            actLoadState[0]->setData(QVariant(0)); +            connect(actLoadState[0], &QAction::triggered, this, &MainWindow::onLoadState); +        } + +        actUndoStateLoad = menu->addAction("Undo state load"); +        actUndoStateLoad->setShortcut(QKeySequence(Qt::Key_F12)); +        connect(actUndoStateLoad, &QAction::triggered, this, &MainWindow::onUndoStateLoad); + +        menu->addSeparator(); + +        actQuit = menu->addAction("Quit"); +        connect(actQuit, &QAction::triggered, this, &MainWindow::onQuit); +    } +    { +        QMenu* menu = menubar->addMenu("System"); + +        actPause = menu->addAction("Pause"); +        actPause->setCheckable(true); +        connect(actPause, &QAction::triggered, this, &MainWindow::onPause); + +        actReset = menu->addAction("Reset"); +        connect(actReset, &QAction::triggered, this, &MainWindow::onReset); + +        actStop = menu->addAction("Stop"); +        connect(actStop, &QAction::triggered, this, &MainWindow::onStop); +    } +    { +        QMenu* menu = menubar->addMenu("Config"); + +        actEmuSettings = menu->addAction("Emu settings"); +        connect(actEmuSettings, &QAction::triggered, this, &MainWindow::onOpenEmuSettings); + +        actInputConfig = menu->addAction("Input and hotkeys"); +        connect(actInputConfig, &QAction::triggered, this, &MainWindow::onOpenInputConfig); + +        actVideoSettings = menu->addAction("Video settings"); +        connect(actVideoSettings, &QAction::triggered, this, &MainWindow::onOpenVideoSettings); + +        actAudioSettings = menu->addAction("Audio settings"); +        connect(actAudioSettings, &QAction::triggered, this, &MainWindow::onOpenAudioSettings); + +        actWifiSettings = menu->addAction("Wifi settings"); +        connect(actWifiSettings, &QAction::triggered, this, &MainWindow::onOpenWifiSettings); + +        { +            QMenu* submenu = menu->addMenu("Savestate settings"); + +            actSavestateSRAMReloc = submenu->addAction("Separate savefiles"); +            actSavestateSRAMReloc->setCheckable(true); +            connect(actSavestateSRAMReloc, &QAction::triggered, this, &MainWindow::onChangeSavestateSRAMReloc); +        } + +        menu->addSeparator(); + +        { +            QMenu* submenu = menu->addMenu("Screen size"); + +            for (int i = 0; i < 4; i++) +            { +                int data = i+1; +                actScreenSize[i] = submenu->addAction(QString("%1x").arg(data)); +                actScreenSize[i]->setData(QVariant(data)); +                connect(actScreenSize[i], &QAction::triggered, this, &MainWindow::onChangeScreenSize); +            } +        } +        { +            QMenu* submenu = menu->addMenu("Screen rotation"); +            grpScreenRotation = new QActionGroup(submenu); + +            for (int i = 0; i < 4; i++) +            { +                int data = i*90; +                actScreenRotation[i] = submenu->addAction(QString("%1°").arg(data)); +                actScreenRotation[i]->setActionGroup(grpScreenRotation); +                actScreenRotation[i]->setData(QVariant(i)); +                actScreenRotation[i]->setCheckable(true); +            } + +            connect(grpScreenRotation, &QActionGroup::triggered, this, &MainWindow::onChangeScreenRotation); +        } +        { +            QMenu* submenu = menu->addMenu("Screen gap"); +            grpScreenGap = new QActionGroup(submenu); + +            const int screengap[] = {0, 1, 8, 64, 90, 128}; + +            for (int i = 0; i < 6; i++) +            { +                int data = screengap[i]; +                actScreenGap[i] = submenu->addAction(QString("%1 px").arg(data)); +                actScreenGap[i]->setActionGroup(grpScreenGap); +                actScreenGap[i]->setData(QVariant(data)); +                actScreenGap[i]->setCheckable(true); +            } + +            connect(grpScreenGap, &QActionGroup::triggered, this, &MainWindow::onChangeScreenGap); +        } +        { +            QMenu* submenu = menu->addMenu("Screen layout"); +            grpScreenLayout = new QActionGroup(submenu); + +            const char* screenlayout[] = {"Natural", "Vertical", "Horizontal"}; + +            for (int i = 0; i < 3; i++) +            { +                actScreenLayout[i] = submenu->addAction(QString(screenlayout[i])); +                actScreenLayout[i]->setActionGroup(grpScreenLayout); +                actScreenLayout[i]->setData(QVariant(i)); +                actScreenLayout[i]->setCheckable(true); +            } + +            connect(grpScreenLayout, &QActionGroup::triggered, this, &MainWindow::onChangeScreenLayout); +        } +        { +            QMenu* submenu = menu->addMenu("Screen sizing"); +            grpScreenSizing = new QActionGroup(submenu); + +            const char* screensizing[] = {"Even", "Emphasize top", "Emphasize bottom", "Auto"}; + +            for (int i = 0; i < 4; i++) +            { +                actScreenSizing[i] = submenu->addAction(QString(screensizing[i])); +                actScreenSizing[i]->setActionGroup(grpScreenSizing); +                actScreenSizing[i]->setData(QVariant(i)); +                actScreenSizing[i]->setCheckable(true); +            } + +            connect(grpScreenSizing, &QActionGroup::triggered, this, &MainWindow::onChangeScreenSizing); + +            submenu->addSeparator(); + +            actIntegerScaling = submenu->addAction("Force integer scaling"); +            actIntegerScaling->setCheckable(true); +            connect(actIntegerScaling, &QAction::triggered, this, &MainWindow::onChangeIntegerScaling); +        } + +        actScreenFiltering = menu->addAction("Screen filtering"); +        actScreenFiltering->setCheckable(true); +        connect(actScreenFiltering, &QAction::triggered, this, &MainWindow::onChangeScreenFiltering); + +        actShowOSD = menu->addAction("Show OSD"); +        actShowOSD->setCheckable(true); +        connect(actShowOSD, &QAction::triggered, this, &MainWindow::onChangeShowOSD); + +        menu->addSeparator(); + +        actLimitFramerate = menu->addAction("Limit framerate"); +        actLimitFramerate->setCheckable(true); +        connect(actLimitFramerate, &QAction::triggered, this, &MainWindow::onChangeLimitFramerate); + +        actAudioSync = menu->addAction("Audio sync"); +        actAudioSync->setCheckable(true); +        connect(actAudioSync, &QAction::triggered, this, &MainWindow::onChangeAudioSync); +    } +    setMenuBar(menubar); + +    resize(Config::WindowWidth, Config::WindowHeight); + +    show(); +    createScreenPanel(); + +    for (int i = 0; i < 9; i++) +    { +        actSaveState[i]->setEnabled(false); +        actLoadState[i]->setEnabled(false); +    } +    actUndoStateLoad->setEnabled(false); + +    actPause->setEnabled(false); +    actReset->setEnabled(false); +    actStop->setEnabled(false); + + +    actSavestateSRAMReloc->setChecked(Config::SavestateRelocSRAM != 0); + +    actScreenRotation[Config::ScreenRotation]->setChecked(true); + +    for (int i = 0; i < 6; i++) +    { +        if (actScreenGap[i]->data().toInt() == Config::ScreenGap) +        { +            actScreenGap[i]->setChecked(true); +            break; +        } +    } + +    actScreenLayout[Config::ScreenLayout]->setChecked(true); +    actScreenSizing[Config::ScreenSizing]->setChecked(true); +    actIntegerScaling->setChecked(Config::IntegerScaling != 0); + +    actScreenFiltering->setChecked(Config::ScreenFilter != 0); +    actShowOSD->setChecked(Config::ShowOSD != 0); + +    actLimitFramerate->setChecked(Config::LimitFPS != 0); +    actAudioSync->setChecked(Config::AudioSync != 0); +} + +MainWindow::~MainWindow() +{ +} + +void MainWindow::createScreenPanel() +{ +    hasOGL = (Config::ScreenUseGL != 0) || (Config::_3DRenderer != 0); + +    if (hasOGL) +    { +        ScreenPanelGL* panelGL = new ScreenPanelGL(this); +        panelGL->show(); + +        if (!panelGL->isValid()) +            hasOGL = false; +        else +        { +            QSurfaceFormat fmt = panelGL->format(); +            if (fmt.majorVersion() < 3 || (fmt.majorVersion() == 3 && fmt.minorVersion() < 2)) +                hasOGL = false; +        } + +        if (!hasOGL) +            delete panelGL; +        else +            panel = panelGL; +    } + +    if (!hasOGL) +    { +        panel = new ScreenPanelNative(this); +        panel->show(); +    } + +    setCentralWidget(panel); +    connect(this, SIGNAL(screenLayoutChange()), panel, SLOT(onScreenLayoutChanged())); +    emit screenLayoutChange(); +} + +QOpenGLContext* MainWindow::getOGLContext() +{ +    if (!hasOGL) return nullptr; + +    QOpenGLWidget* glpanel = (QOpenGLWidget*)panel; +    return glpanel->context(); +} + +void MainWindow::resizeEvent(QResizeEvent* event) +{ +    int w = event->size().width(); +    int h = event->size().height(); + +    Config::WindowWidth = w; +    Config::WindowHeight = h; + +    // TODO: detect when the window gets maximized! +} + +void MainWindow::keyPressEvent(QKeyEvent* event) +{ +    if (event->isAutoRepeat()) return; + +    Input::KeyPress(event); +} + +void MainWindow::keyReleaseEvent(QKeyEvent* event) +{ +    if (event->isAutoRepeat()) return; + +    Input::KeyRelease(event); +} + + +void MainWindow::dragEnterEvent(QDragEnterEvent* event) +{ +    if (!event->mimeData()->hasUrls()) return; + +    QList<QUrl> urls = event->mimeData()->urls(); +    if (urls.count() > 1) return; // not handling more than one file at once + +    QString filename = urls.at(0).toLocalFile(); +    QString ext = filename.right(3); + +    if (ext == "nds" || ext == "srl" || (ext == "gba" && RunningSomething)) +        event->acceptProposedAction(); +} + +void MainWindow::dropEvent(QDropEvent* event) +{ +    if (!event->mimeData()->hasUrls()) return; + +    QList<QUrl> urls = event->mimeData()->urls(); +    if (urls.count() > 1) return; // not handling more than one file at once + +    emuThread->emuPause(); + +    QString filename = urls.at(0).toLocalFile(); +    QString ext = filename.right(3); + +    char _filename[1024]; +    strncpy(_filename, filename.toStdString().c_str(), 1023); _filename[1023] = '\0'; + +    int slot; int res; +    if (ext == "gba") +    { +        slot = 1; +        res = Frontend::LoadROM(_filename, Frontend::ROMSlot_GBA); +    } +    else +    { +        slot = 0; +        res = Frontend::LoadROM(_filename, Frontend::ROMSlot_NDS); +    } + +    if (res != Frontend::Load_OK) +    { +        QMessageBox::critical(this, +                              "melonDS", +                              loadErrorStr(res)); +        emuThread->emuUnpause(); +    } +    else if (slot == 1) +    { +        // checkme +        emuThread->emuUnpause(); +    } +    else +    { +        emuThread->emuRun(); +    } +} + + +QString MainWindow::loadErrorStr(int error) +{ +    switch (error) +    { +    case Frontend::Load_BIOS9Missing: return "DS ARM9 BIOS was not found or could not be accessed."; +    case Frontend::Load_BIOS9Bad:     return "DS ARM9 BIOS is not a valid BIOS dump."; + +    case Frontend::Load_BIOS7Missing: return "DS ARM7 BIOS was not found or could not be accessed."; +    case Frontend::Load_BIOS7Bad:     return "DS ARM7 BIOS is not a valid BIOS dump."; + +    case Frontend::Load_FirmwareMissing:     return "DS firmware was not found or could not be accessed."; +    case Frontend::Load_FirmwareBad:         return "DS firmware is not a valid firmware dump."; +    case Frontend::Load_FirmwareNotBootable: return "DS firmware is not bootable."; + +    case Frontend::Load_ROMLoadError: return "Failed to load the ROM. Make sure the file is accessible and isn't used by another application."; + +    default: return "Unknown error during launch; smack Arisotura."; +    } +} + + +void MainWindow::onOpenFile() +{ +    emuThread->emuPause(); + +    QString filename = QFileDialog::getOpenFileName(this, +                                                    "Open ROM", +                                                    Config::LastROMFolder, +                                                    "DS ROMs (*.nds *.srl);;GBA ROMs (*.gba);;Any file (*.*)"); +    if (filename.isEmpty()) +    { +        emuThread->emuUnpause(); +        return; +    } + +    // TODO: validate the input file!! +    // * check that it is a proper ROM +    // * ensure the binary offsets are sane +    // * etc + +    // this shit is stupid +    char file[1024]; +    strncpy(file, filename.toStdString().c_str(), 1023); file[1023] = '\0'; + +    int pos = strlen(file)-1; +    while (file[pos] != '/' && file[pos] != '\\' && pos > 0) pos--; +    strncpy(Config::LastROMFolder, file, pos); +    Config::LastROMFolder[pos] = '\0'; +    char* ext = &file[strlen(file)-3]; + +    int slot; int res; +    if (!strcasecmp(ext, "gba")) +    { +        slot = 1; +        res = Frontend::LoadROM(file, Frontend::ROMSlot_GBA); +    } +    else +    { +        slot = 0; +        res = Frontend::LoadROM(file, Frontend::ROMSlot_NDS); +    } + +    if (res != Frontend::Load_OK) +    { +        QMessageBox::critical(this, +                              "melonDS", +                              loadErrorStr(res)); +        emuThread->emuUnpause(); +    } +    else if (slot == 1) +    { +        // checkme +        emuThread->emuUnpause(); +    } +    else +    { +        emuThread->emuRun(); +    } +} + +void MainWindow::onBootFirmware() +{ +    // TODO: check the whole GBA cart shito + +    emuThread->emuPause(); + +    int res = Frontend::LoadBIOS(); +    if (res != Frontend::Load_OK) +    { +        QMessageBox::critical(this, +                              "melonDS", +                              loadErrorStr(res)); +        emuThread->emuUnpause(); +    } +    else +    { +        emuThread->emuRun(); +    } +} + +void MainWindow::onSaveState() +{ +    int slot = ((QAction*)sender())->data().toInt(); + +    emuThread->emuPause(); + +    char filename[1024]; +    if (slot > 0) +    { +        Frontend::GetSavestateName(slot, filename, 1024); +    } +    else +    { +        // TODO: specific 'last directory' for savestate files? +        QString qfilename = QFileDialog::getSaveFileName(this, +                                                         "Save state", +                                                         Config::LastROMFolder, +                                                         "melonDS savestates (*.mln);;Any file (*.*)"); +        if (qfilename.isEmpty()) +        { +            emuThread->emuUnpause(); +            return; +        } + +        strncpy(filename, qfilename.toStdString().c_str(), 1023); filename[1023] = '\0'; +    } + +    if (Frontend::SaveState(filename)) +    { +        char msg[64]; +        if (slot > 0) sprintf(msg, "State saved to slot %d", slot); +        else          sprintf(msg, "State saved to file"); +        OSD::AddMessage(0, msg); + +        actLoadState[slot]->setEnabled(true); +    } +    else +    { +        OSD::AddMessage(0xFFA0A0, "State save failed"); +    } + +    emuThread->emuUnpause(); +} + +void MainWindow::onLoadState() +{ +    int slot = ((QAction*)sender())->data().toInt(); + +    emuThread->emuPause(); + +    char filename[1024]; +    if (slot > 0) +    { +        Frontend::GetSavestateName(slot, filename, 1024); +    } +    else +    { +        // TODO: specific 'last directory' for savestate files? +        QString qfilename = QFileDialog::getOpenFileName(this, +                                                         "Load state", +                                                         Config::LastROMFolder, +                                                         "melonDS savestates (*.ml*);;Any file (*.*)"); +        if (qfilename.isEmpty()) +        { +            emuThread->emuUnpause(); +            return; +        } + +        strncpy(filename, qfilename.toStdString().c_str(), 1023); filename[1023] = '\0'; +    } + +    if (!Platform::FileExists(filename)) +    { +        char msg[64]; +        if (slot > 0) sprintf(msg, "State slot %d is empty", slot); +        else          sprintf(msg, "State file does not exist"); +        OSD::AddMessage(0xFFA0A0, msg); + +        emuThread->emuUnpause(); +        return; +    } + +    if (Frontend::LoadState(filename)) +    { +        char msg[64]; +        if (slot > 0) sprintf(msg, "State loaded from slot %d", slot); +        else          sprintf(msg, "State loaded from file"); +        OSD::AddMessage(0, msg); +    } +    else +    { +        OSD::AddMessage(0xFFA0A0, "State load failed"); +    } + +    emuThread->emuUnpause(); +} + +void MainWindow::onUndoStateLoad() +{ +    emuThread->emuPause(); +    Frontend::UndoStateLoad(); +    emuThread->emuUnpause(); + +    OSD::AddMessage(0, "State load undone"); +} + +void MainWindow::onQuit() +{ +    QApplication::quit(); +} + + +void MainWindow::onPause(bool checked) +{ +    if (!RunningSomething) return; + +    if (checked) +    { +        emuThread->emuPause(); +        OSD::AddMessage(0, "Paused"); +    } +    else +    { +        emuThread->emuUnpause(); +        OSD::AddMessage(0, "Resumed"); +    } +} + +void MainWindow::onReset() +{ +    if (!RunningSomething) return; + +    emuThread->emuPause(); + +    actUndoStateLoad->setEnabled(false); + +    int res = Frontend::Reset(); +    if (res != Frontend::Load_OK) +    { +        QMessageBox::critical(this, +                              "melonDS", +                              loadErrorStr(res)); +        emuThread->emuUnpause(); +    } +    else +    { +        OSD::AddMessage(0, "Reset"); +        emuThread->emuRun(); +    } +} + +void MainWindow::onStop() +{ +    if (!RunningSomething) return; + +    emuThread->emuPause(); +    NDS::Stop(); +} + + +void MainWindow::onOpenEmuSettings() +{ +    EmuSettingsDialog::openDlg(this); +} + +void MainWindow::onOpenInputConfig() +{ +    emuThread->emuPause(); + +    InputConfigDialog* dlg = InputConfigDialog::openDlg(this); +    connect(dlg, &InputConfigDialog::finished, this, &MainWindow::onInputConfigFinished); +} + +void MainWindow::onInputConfigFinished(int res) +{ +    emuThread->emuUnpause(); +} + +void MainWindow::onOpenVideoSettings() +{ +    VideoSettingsDialog* dlg = VideoSettingsDialog::openDlg(this); +    connect(dlg, &VideoSettingsDialog::updateVideoSettings, this, &MainWindow::onUpdateVideoSettings); +} + +void MainWindow::onOpenAudioSettings() +{ +    AudioSettingsDialog* dlg = AudioSettingsDialog::openDlg(this); +    connect(dlg, &AudioSettingsDialog::finished, this, &MainWindow::onAudioSettingsFinished); +} + +void MainWindow::onAudioSettingsFinished(int res) +{ +    if (Config::MicInputType == 3) +    { +        micLoadWav(Config::MicWavPath); +        Frontend::Mic_SetExternalBuffer(micWavBuffer, micWavLength); +    } +    else +    { +        delete[] micWavBuffer; +        micWavBuffer = nullptr; + +        if (Config::MicInputType == 1) +            Frontend::Mic_SetExternalBuffer(micExtBuffer, sizeof(micExtBuffer)/sizeof(s16)); +        else +            Frontend::Mic_SetExternalBuffer(NULL, 0); +    } +} + +void MainWindow::onOpenWifiSettings() +{ +    WifiSettingsDialog* dlg = WifiSettingsDialog::openDlg(this); +    connect(dlg, &WifiSettingsDialog::finished, this, &MainWindow::onWifiSettingsFinished); +} + +void MainWindow::onWifiSettingsFinished(int res) +{ +    emuThread->emuPause(); + +    if (Wifi::MPInited) +    { +        Platform::MP_DeInit(); +        Platform::MP_Init(); +    } + +    Platform::LAN_DeInit(); +    Platform::LAN_Init(); + +    emuThread->emuUnpause(); +} + +void MainWindow::onChangeSavestateSRAMReloc(bool checked) +{ +    Config::SavestateRelocSRAM = checked?1:0; +} + +void MainWindow::onChangeScreenSize() +{ +    int factor = ((QAction*)sender())->data().toInt(); + +    bool isHori = (Config::ScreenRotation == 1 || Config::ScreenRotation == 3); +    int gap = Config::ScreenGap; + +    int w = 256*factor; +    int h = 192*factor; + +    QSize diff = size() - panel->size(); + +    if (Config::ScreenLayout == 0) // natural +    { +        if (isHori) +            resize(QSize(h+gap+h, w) + diff); +        else +            resize(QSize(w, h+gap+h) + diff); +    } +    else if (Config::ScreenLayout == 1) // vertical +    { +        if (isHori) +            resize(QSize(h, w+gap+w) + diff); +        else +            resize(QSize(w, h+gap+h) + diff); +    } +    else // horizontal +    { +        if (isHori) +            resize(QSize(h+gap+h, w) + diff); +        else +            resize(QSize(w+gap+w, h) + diff); +    } +} + +void MainWindow::onChangeScreenRotation(QAction* act) +{ +    int rot = act->data().toInt(); +    Config::ScreenRotation = rot; + +    emit screenLayoutChange(); +} + +void MainWindow::onChangeScreenGap(QAction* act) +{ +    int gap = act->data().toInt(); +    Config::ScreenGap = gap; + +    emit screenLayoutChange(); +} + +void MainWindow::onChangeScreenLayout(QAction* act) +{ +    int layout = act->data().toInt(); +    Config::ScreenLayout = layout; + +    emit screenLayoutChange(); +} + +void MainWindow::onChangeScreenSizing(QAction* act) +{ +    int sizing = act->data().toInt(); +    Config::ScreenSizing = sizing; + +    emit screenLayoutChange(); +} + +void MainWindow::onChangeIntegerScaling(bool checked) +{ +    Config::IntegerScaling = checked?1:0; + +    emit screenLayoutChange(); +} + +void MainWindow::onChangeScreenFiltering(bool checked) +{ +    Config::ScreenFilter = checked?1:0; +} + +void MainWindow::onChangeShowOSD(bool checked) +{ +    Config::ShowOSD = checked?1:0; +} + +void MainWindow::onChangeLimitFramerate(bool checked) +{ +    Config::LimitFPS = checked?1:0; +} + +void MainWindow::onChangeAudioSync(bool checked) +{ +    Config::AudioSync = checked?1:0; +} + + +void MainWindow::onTitleUpdate(QString title) +{ +    setWindowTitle(title); +} + +void MainWindow::onEmuStart() +{ +    for (int i = 1; i < 9; i++) +    { +        actSaveState[i]->setEnabled(true); +        actLoadState[i]->setEnabled(Frontend::SavestateExists(i)); +    } +    actSaveState[0]->setEnabled(true); +    actLoadState[0]->setEnabled(true); +    actUndoStateLoad->setEnabled(false); + +    actPause->setEnabled(true); +    actPause->setChecked(false); +    actReset->setEnabled(true); +    actStop->setEnabled(true); +} + +void MainWindow::onEmuStop() +{ +    emuThread->emuPause(); + +    for (int i = 0; i < 9; i++) +    { +        actSaveState[i]->setEnabled(false); +        actLoadState[i]->setEnabled(false); +    } +    actUndoStateLoad->setEnabled(false); + +    actPause->setEnabled(false); +    actReset->setEnabled(false); +    actStop->setEnabled(false); +} + +void MainWindow::onUpdateVideoSettings(bool glchange) +{ +    if (glchange) +    { +        emuThread->emuPause(); + +        if (hasOGL) emuThread->deinitOpenGL(); +        delete panel; +        createScreenPanel(); +        connect(emuThread, SIGNAL(windowUpdate()), panel, SLOT(update())); +        if (hasOGL) emuThread->initOpenGL(); +    } + +    videoSettingsDirty = true; + +    if (glchange) +        emuThread->emuUnpause(); +} + + +void emuStop() +{ +    RunningSomething = false; + +    Frontend::UnloadROM(Frontend::ROMSlot_NDS); +    Frontend::UnloadROM(Frontend::ROMSlot_GBA); + +    emit emuThread->windowEmuStop(); + +    OSD::AddMessage(0xFFC040, "Shutdown"); +} + + + +int main(int argc, char** argv) +{ +    srand(time(NULL)); + +    printf("melonDS " MELONDS_VERSION "\n"); +    printf(MELONDS_URL "\n"); + +    Platform::Init(argc, argv); + +    QApplication melon(argc, argv); +    melon.setWindowIcon(QIcon(":/melon-icon")); + +    // http://stackoverflow.com/questions/14543333/joystick-wont-work-using-sdl +    SDL_SetHint(SDL_HINT_JOYSTICK_ALLOW_BACKGROUND_EVENTS, "1"); + +    if (SDL_Init(SDL_INIT_HAPTIC) < 0) +    { +        printf("SDL couldn't init rumble\n"); +    } +    if (SDL_Init(SDL_INIT_VIDEO | SDL_INIT_AUDIO | SDL_INIT_JOYSTICK) < 0) +    { +        QMessageBox::critical(NULL, "melonDS", "SDL shat itself :("); +        return 1; +    } + +    SDL_JoystickEventState(SDL_ENABLE); + +    Config::Load(); + +#define SANITIZE(var, min, max)  { if (var < min) var = min; else if (var > max) var = max; } +    SANITIZE(Config::_3DRenderer, 0, 1); +    SANITIZE(Config::ScreenVSyncInterval, 1, 20); +    SANITIZE(Config::GL_ScaleFactor, 1, 16); +    SANITIZE(Config::AudioVolume, 0, 256); +    SANITIZE(Config::MicInputType, 0, 3); +    SANITIZE(Config::ScreenRotation, 0, 3); +    SANITIZE(Config::ScreenGap, 0, 500); +    SANITIZE(Config::ScreenLayout, 0, 2); +    SANITIZE(Config::ScreenSizing, 0, 3); +#undef SANITIZE + +    // TODO: this should be checked before running anything +#if 0 +    { +        const char* romlist_missing = "Save memory type detection will not work correctly.\n\n" +            "You should use the latest version of romlist.bin (provided in melonDS release packages)."; +#if !defined(UNIX_PORTABLE) && !defined(__WIN32__) +        std::string missingstr = std::string(romlist_missing) + +            "\n\nThe ROM list should be placed in " + g_get_user_data_dir() + "/melonds/, otherwise " +            "melonDS will search for it in the current working directory."; +        const char* romlist_missing_text = missingstr.c_str(); +#else +        const char* romlist_missing_text = romlist_missing; +#endif + +        FILE* f = Platform::OpenDataFile("romlist.bin"); +        if (f) +        { +            u32 data; +            fread(&data, 4, 1, f); +            fclose(f); + +            if ((data >> 24) == 0) // old CRC-based list +            { +                uiMsgBoxError(NULL, "Your version of romlist.bin is outdated.", romlist_missing_text); +            } +        } +        else +        { +        	uiMsgBoxError(NULL, "romlist.bin not found.", romlist_missing_text); +        } +    } +#endif + +    QSurfaceFormat format; +    format.setDepthBufferSize(24); +    format.setStencilBufferSize(8); +    format.setVersion(3, 2); +    format.setProfile(QSurfaceFormat::CoreProfile); +    format.setSwapInterval(0); +    QSurfaceFormat::setDefaultFormat(format); + +    audioSync = SDL_CreateCond(); +    audioSyncLock = SDL_CreateMutex(); + +    audioFreq = 48000; // TODO: make configurable? +    SDL_AudioSpec whatIwant, whatIget; +    memset(&whatIwant, 0, sizeof(SDL_AudioSpec)); +    whatIwant.freq = audioFreq; +    whatIwant.format = AUDIO_S16LSB; +    whatIwant.channels = 2; +    whatIwant.samples = 1024; +    whatIwant.callback = audioCallback; +    audioDevice = SDL_OpenAudioDevice(NULL, 0, &whatIwant, &whatIget, SDL_AUDIO_ALLOW_FREQUENCY_CHANGE); +    if (!audioDevice) +    { +        printf("Audio init failed: %s\n", SDL_GetError()); +    } +    else +    { +        audioFreq = whatIget.freq; +        printf("Audio output frequency: %d Hz\n", audioFreq); +        SDL_PauseAudioDevice(audioDevice, 1); +    } + +    memset(&whatIwant, 0, sizeof(SDL_AudioSpec)); +    whatIwant.freq = 44100; +    whatIwant.format = AUDIO_S16LSB; +    whatIwant.channels = 1; +    whatIwant.samples = 1024; +    whatIwant.callback = micCallback; +    micDevice = SDL_OpenAudioDevice(NULL, 1, &whatIwant, &whatIget, 0); +    if (!micDevice) +    { +        printf("Mic init failed: %s\n", SDL_GetError()); +    } +    else +    { +        SDL_PauseAudioDevice(micDevice, 1); +    } + + +    memset(micExtBuffer, 0, sizeof(micExtBuffer)); +    micExtBufferWritePos = 0; +    micWavBuffer = nullptr; + +    Frontend::Init_ROM(); +    Frontend::Init_Audio(audioFreq); + +    if (Config::MicInputType == 1) +    { +        Frontend::Mic_SetExternalBuffer(micExtBuffer, sizeof(micExtBuffer)/sizeof(s16)); +    } +    else if (Config::MicInputType == 3) +    { +        micLoadWav(Config::MicWavPath); +        Frontend::Mic_SetExternalBuffer(micWavBuffer, micWavLength); +    } + +    Input::JoystickID = Config::JoystickID; +    Input::OpenJoystick(); + +    mainWindow = new MainWindow(); + +    emuThread = new EmuThread(); +    emuThread->start(); +    emuThread->emuPause(); + +    if (argc > 1) +    { +        char* file = argv[1]; +        char* ext = &file[strlen(file)-3]; + +        if (!strcasecmp(ext, "nds") || !strcasecmp(ext, "srl")) +        { +            int res = Frontend::LoadROM(file, Frontend::ROMSlot_NDS); + +            if (res == Frontend::Load_OK) +            { +                if (argc > 2) +                { +                    file = argv[2]; +                    ext = &file[strlen(file)-3]; + +                    if (!strcasecmp(ext, "gba")) +                    { +                        Frontend::LoadROM(file, Frontend::ROMSlot_GBA); +                    } +                } + +                emuThread->emuRun(); +            } +        } +    } + +    int ret = melon.exec(); + +    emuThread->emuStop(); +    emuThread->wait(); +    delete emuThread; + +    Input::CloseJoystick(); + +    if (audioDevice) SDL_CloseAudioDevice(audioDevice); +    if (micDevice)   SDL_CloseAudioDevice(micDevice); + +    SDL_DestroyCond(audioSync); +    SDL_DestroyMutex(audioSyncLock); + +    if (micWavBuffer) delete[] micWavBuffer; + +    Config::Save(); + +    SDL_Quit(); +    Platform::DeInit(); +    return ret; +} + +#ifdef __WIN32__ + +#include <windows.h> + +int CALLBACK WinMain(HINSTANCE hinst, HINSTANCE hprev, LPSTR cmdline, int cmdshow) +{ +    int argc = 0; +    wchar_t** argv_w = CommandLineToArgvW(GetCommandLineW(), &argc); +    char* nullarg = ""; + +    char** argv = new char*[argc]; +    for (int i = 0; i < argc; i++) +    { +        if (!argv_w) { argv[i] = nullarg; continue; } +        int len = WideCharToMultiByte(CP_UTF8, 0, argv_w[i], -1, NULL, 0, NULL, NULL); +        if (len < 1) { argv[i] = nullarg; continue; } +        argv[i] = new char[len]; +        int res = WideCharToMultiByte(CP_UTF8, 0, argv_w[i], -1, argv[i], len, NULL, NULL); +        if (res != len) { delete[] argv[i]; argv[i] = nullarg; } +    } + +    if (argv_w) LocalFree(argv_w); + +    if (AttachConsole(ATTACH_PARENT_PROCESS)) +    { +        freopen("CONOUT$", "w", stdout); +        freopen("CONOUT$", "w", stderr); +        printf("\n"); +    } + +    int ret = main(argc, argv); + +    printf("\n\n>"); + +    for (int i = 0; i < argc; i++) if (argv[i] != nullarg) delete[] argv[i]; +    delete[] argv; + +    return ret; +} + +#endif diff --git a/src/frontend/qt_sdl/main.h b/src/frontend/qt_sdl/main.h new file mode 100644 index 0000000..279aed8 --- /dev/null +++ b/src/frontend/qt_sdl/main.h @@ -0,0 +1,269 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef MAIN_H +#define MAIN_H + +#include <QThread> +#include <QWidget> +#include <QWindow> +#include <QMainWindow> +#include <QImage> +#include <QActionGroup> + +#include <QOffscreenSurface> +#include <QOpenGLWidget> +#include <QOpenGLContext> +#include <QOpenGLFunctions> +#include <QOpenGLFunctions_3_2_Core> +#include <QOpenGLShaderProgram> + + +class EmuThread : public QThread +{ +    Q_OBJECT +    void run() override; + +public: +    explicit EmuThread(QObject* parent = nullptr); + +    void initOpenGL(); +    void deinitOpenGL(); + +    void* oglGetProcAddress(const char* proc); + +    void changeWindowTitle(char* title); + +    // to be called from the UI thread +    void emuRun(); +    void emuPause(); +    void emuUnpause(); +    void emuStop(); + +    bool emuIsRunning(); + +signals: +    void windowUpdate(); +    void windowTitleChange(QString title); + +    void windowEmuStart(); +    void windowEmuStop(); +    void windowEmuPause(); +    void windowEmuReset(); + +    void windowLimitFPSChange(); + +    void screenLayoutChange(); + +private: +    volatile int EmuStatus; +    int PrevEmuStatus; +    int EmuRunning; + +    QOffscreenSurface* oglSurface; +    QOpenGLContext* oglContext; +}; + + +class ScreenHandler +{ +    Q_GADGET + +public: +    virtual ~ScreenHandler() {} + +protected: +    void screenSetupLayout(int w, int h); + +    QSize screenGetMinSize(); + +    void screenOnMousePress(QMouseEvent* event); +    void screenOnMouseRelease(QMouseEvent* event); +    void screenOnMouseMove(QMouseEvent* event); + +    float screenMatrix[2][6]; + +    bool touching; +}; + + +class ScreenPanelNative : public QWidget, public ScreenHandler +{ +    Q_OBJECT + +public: +    explicit ScreenPanelNative(QWidget* parent); +    ~ScreenPanelNative(); + +protected: +    void paintEvent(QPaintEvent* event) override; + +    void resizeEvent(QResizeEvent* event) override; + +    void mousePressEvent(QMouseEvent* event) override; +    void mouseReleaseEvent(QMouseEvent* event) override; +    void mouseMoveEvent(QMouseEvent* event) override; + +private slots: +    void onScreenLayoutChanged(); + +private: +    void setupScreenLayout(); + +    QImage screen[2]; +    QTransform screenTrans[2]; +}; + + +class ScreenPanelGL : public QOpenGLWidget, public ScreenHandler, protected QOpenGLFunctions_3_2_Core +{ +    Q_OBJECT + +public: +    explicit ScreenPanelGL(QWidget* parent); +    ~ScreenPanelGL(); + +protected: +    void initializeGL() override; + +    void paintGL() override; + +    void resizeEvent(QResizeEvent* event) override; +    void resizeGL(int w, int h) override; + +    void mousePressEvent(QMouseEvent* event) override; +    void mouseReleaseEvent(QMouseEvent* event) override; +    void mouseMoveEvent(QMouseEvent* event) override; + +private slots: +    void onScreenLayoutChanged(); + +private: +    void setupScreenLayout(); + +    QOpenGLShaderProgram* screenShader; +    GLuint screenVertexBuffer; +    GLuint screenVertexArray; +    GLuint screenTexture; +}; + + +class MainWindow : public QMainWindow +{ +    Q_OBJECT + +public: +    explicit MainWindow(QWidget* parent = nullptr); +    ~MainWindow(); + +    bool hasOGL; +    QOpenGLContext* getOGLContext(); + +protected: +    void resizeEvent(QResizeEvent* event) override; + +    void keyPressEvent(QKeyEvent* event) override; +    void keyReleaseEvent(QKeyEvent* event) override; + +    void dragEnterEvent(QDragEnterEvent* event) override; +    void dropEvent(QDropEvent* event) override; + +signals: +    void screenLayoutChange(); + +private slots: +    void onOpenFile(); +    void onBootFirmware(); +    void onSaveState(); +    void onLoadState(); +    void onUndoStateLoad(); +    void onQuit(); + +    void onPause(bool checked); +    void onReset(); +    void onStop(); + +    void onOpenEmuSettings(); +    void onOpenInputConfig(); +    void onInputConfigFinished(int res); +    void onOpenVideoSettings(); +    void onOpenAudioSettings(); +    void onAudioSettingsFinished(int res); +    void onOpenWifiSettings(); +    void onWifiSettingsFinished(int res); +    void onChangeSavestateSRAMReloc(bool checked); +    void onChangeScreenSize(); +    void onChangeScreenRotation(QAction* act); +    void onChangeScreenGap(QAction* act); +    void onChangeScreenLayout(QAction* act); +    void onChangeScreenSizing(QAction* act); +    void onChangeIntegerScaling(bool checked); +    void onChangeScreenFiltering(bool checked); +    void onChangeShowOSD(bool checked); +    void onChangeLimitFramerate(bool checked); +    void onChangeAudioSync(bool checked); + +    void onTitleUpdate(QString title); + +    void onEmuStart(); +    void onEmuStop(); + +    void onUpdateVideoSettings(bool glchange); + +private: +    void createScreenPanel(); + +    QString loadErrorStr(int error); + +public: +    QWidget* panel; + +    QAction* actOpenROM; +    QAction* actBootFirmware; +    QAction* actSaveState[9]; +    QAction* actLoadState[9]; +    QAction* actUndoStateLoad; +    QAction* actQuit; + +    QAction* actPause; +    QAction* actReset; +    QAction* actStop; + +    QAction* actEmuSettings; +    QAction* actInputConfig; +    QAction* actVideoSettings; +    QAction* actAudioSettings; +    QAction* actWifiSettings; +    QAction* actSavestateSRAMReloc; +    QAction* actScreenSize[4]; +    QActionGroup* grpScreenRotation; +    QAction* actScreenRotation[4]; +    QActionGroup* grpScreenGap; +    QAction* actScreenGap[6]; +    QActionGroup* grpScreenLayout; +    QAction* actScreenLayout[3]; +    QActionGroup* grpScreenSizing; +    QAction* actScreenSizing[4]; +    QAction* actIntegerScaling; +    QAction* actScreenFiltering; +    QAction* actShowOSD; +    QAction* actLimitFramerate; +    QAction* actAudioSync; +}; + +#endif // MAIN_H diff --git a/src/frontend/qt_sdl/main_shaders.h b/src/frontend/qt_sdl/main_shaders.h new file mode 100644 index 0000000..c55f79e --- /dev/null +++ b/src/frontend/qt_sdl/main_shaders.h @@ -0,0 +1,64 @@ +/* +    Copyright 2016-2020 Arisotura + +    This file is part of melonDS. + +    melonDS 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. + +    melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. +*/ + +#ifndef MAIN_SHADERS_H +#define MAIN_SHADERS_H + +const char* kScreenVS = R"(#version 140 + +uniform vec2 uScreenSize; +uniform mat2x3 uTransform; + +in vec2 vPosition; +in vec2 vTexcoord; + +smooth out vec2 fTexcoord; + +void main() +{ +    vec4 fpos; + +    fpos.xy = vec3(vPosition, 1.0) * uTransform; + +    fpos.xy = ((fpos.xy * 2.0) / uScreenSize) - 1.0; +    fpos.y *= -1; +    fpos.z = 0.0; +    fpos.w = 1.0; + +    gl_Position = fpos; +    fTexcoord = vTexcoord; +} +)"; + +const char* kScreenFS = R"(#version 140 + +uniform sampler2D ScreenTex; + +smooth in vec2 fTexcoord; + +out vec4 oColor; + +void main() +{ +    vec4 pixel = texture(ScreenTex, fTexcoord); + +    oColor = vec4(pixel.bgr, 1.0); +} +)"; + +#endif // MAIN_SHADERS_H diff --git a/src/frontend/qt_sdl/pcap/bluetooth.h b/src/frontend/qt_sdl/pcap/bluetooth.h new file mode 100644 index 0000000..15dc5a8 --- /dev/null +++ b/src/frontend/qt_sdl/pcap/bluetooth.h @@ -0,0 +1,56 @@ +/* + * Copyright (c) 2006 Paolo Abeni (Italy) + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. The name of the author may not be used to endorse or promote + * products derived from this software without specific prior written + * permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + * bluetooth data struct + * By Paolo Abeni <paolo.abeni@email.it> + */ + +#ifndef lib_pcap_bluetooth_h +#define lib_pcap_bluetooth_h + +#include <pcap/pcap-inttypes.h> + +/* + * Header prepended libpcap to each bluetooth h4 frame, + * fields are in network byte order + */ +typedef struct _pcap_bluetooth_h4_header { +	uint32_t direction; /* if first bit is set direction is incoming */ +} pcap_bluetooth_h4_header; + +/* + * Header prepended libpcap to each bluetooth linux monitor frame, + * fields are in network byte order + */ +typedef struct _pcap_bluetooth_linux_monitor_header { +	uint16_t adapter_id; +	uint16_t opcode; +} pcap_bluetooth_linux_monitor_header; + +#endif diff --git a/src/frontend/qt_sdl/pcap/bpf.h b/src/frontend/qt_sdl/pcap/bpf.h new file mode 100644 index 0000000..1a953a9 --- /dev/null +++ b/src/frontend/qt_sdl/pcap/bpf.h @@ -0,0 +1,270 @@ +/*- + * Copyright (c) 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997 + *	The Regents of the University of California.  All rights reserved. + * + * This code is derived from the Stanford/CMU enet packet filter, + * (net/enet.c) distributed as part of 4.3BSD, and code contributed + * to Berkeley by Steven McCanne and Van Jacobson both of Lawrence + * Berkeley Laboratory. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. Neither the name of the University nor the names of its contributors + *    may be used to endorse or promote products derived from this software + *    without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + * + *      @(#)bpf.h       7.1 (Berkeley) 5/7/91 + */ + +/* + * This is libpcap's cut-down version of bpf.h; it includes only + * the stuff needed for the code generator and the userland BPF + * interpreter, and the libpcap APIs for setting filters, etc.. + * + * "pcap-bpf.c" will include the native OS version, as it deals with + * the OS's BPF implementation. + * + * At least two programs found by Google Code Search explicitly includes + * <pcap/bpf.h> (even though <pcap.h>/<pcap/pcap.h> includes it for you), + * so moving that stuff to <pcap/pcap.h> would break the build for some + * programs. + */ + +/* + * If we've already included <net/bpf.h>, don't re-define this stuff. + * We assume BSD-style multiple-include protection in <net/bpf.h>, + * which is true of all but the oldest versions of FreeBSD and NetBSD, + * or Tru64 UNIX-style multiple-include protection (or, at least, + * Tru64 UNIX 5.x-style; I don't have earlier versions available to check), + * or AIX-style multiple-include protection (or, at least, AIX 5.x-style; + * I don't have earlier versions available to check), or QNX-style + * multiple-include protection (as per GitHub pull request #394). + * + * We do not check for BPF_MAJOR_VERSION, as that's defined by + * <linux/filter.h>, which is directly or indirectly included in some + * programs that also include pcap.h, and <linux/filter.h> doesn't + * define stuff we need. + * + * This also provides our own multiple-include protection. + */ +#if !defined(_NET_BPF_H_) && !defined(_NET_BPF_H_INCLUDED) && !defined(_BPF_H_) && !defined(_H_BPF) && !defined(lib_pcap_bpf_h) +#define lib_pcap_bpf_h + +#include <pcap/funcattrs.h> + +#ifdef __cplusplus +extern "C" { +#endif + +/* BSD style release date */ +#define BPF_RELEASE 199606 + +#ifdef MSDOS /* must be 32-bit */ +typedef long          bpf_int32; +typedef unsigned long bpf_u_int32; +#else +typedef	int bpf_int32; +typedef	u_int bpf_u_int32; +#endif + +/* + * Alignment macros.  BPF_WORDALIGN rounds up to the next + * even multiple of BPF_ALIGNMENT. + * + * Tcpdump's print-pflog.c uses this, so we define it here. + */ +#ifndef __NetBSD__ +#define BPF_ALIGNMENT sizeof(bpf_int32) +#else +#define BPF_ALIGNMENT sizeof(long) +#endif +#define BPF_WORDALIGN(x) (((x)+(BPF_ALIGNMENT-1))&~(BPF_ALIGNMENT-1)) + +/* + * Structure for "pcap_compile()", "pcap_setfilter()", etc.. + */ +struct bpf_program { +	u_int bf_len; +	struct bpf_insn *bf_insns; +}; + +#include <pcap/dlt.h> + +/* + * The instruction encodings. + * + * Please inform tcpdump-workers@lists.tcpdump.org if you use any + * of the reserved values, so that we can note that they're used + * (and perhaps implement it in the reference BPF implementation + * and encourage its implementation elsewhere). + */ + +/* + * The upper 8 bits of the opcode aren't used. BSD/OS used 0x8000. + */ + +/* instruction classes */ +#define BPF_CLASS(code) ((code) & 0x07) +#define		BPF_LD		0x00 +#define		BPF_LDX		0x01 +#define		BPF_ST		0x02 +#define		BPF_STX		0x03 +#define		BPF_ALU		0x04 +#define		BPF_JMP		0x05 +#define		BPF_RET		0x06 +#define		BPF_MISC	0x07 + +/* ld/ldx fields */ +#define BPF_SIZE(code)	((code) & 0x18) +#define		BPF_W		0x00 +#define		BPF_H		0x08 +#define		BPF_B		0x10 +/*				0x18	reserved; used by BSD/OS */ +#define BPF_MODE(code)	((code) & 0xe0) +#define		BPF_IMM 	0x00 +#define		BPF_ABS		0x20 +#define		BPF_IND		0x40 +#define		BPF_MEM		0x60 +#define		BPF_LEN		0x80 +#define		BPF_MSH		0xa0 +/*				0xc0	reserved; used by BSD/OS */ +/*				0xe0	reserved; used by BSD/OS */ + +/* alu/jmp fields */ +#define BPF_OP(code)	((code) & 0xf0) +#define		BPF_ADD		0x00 +#define		BPF_SUB		0x10 +#define		BPF_MUL		0x20 +#define		BPF_DIV		0x30 +#define		BPF_OR		0x40 +#define		BPF_AND		0x50 +#define		BPF_LSH		0x60 +#define		BPF_RSH		0x70 +#define		BPF_NEG		0x80 +#define		BPF_MOD		0x90 +#define		BPF_XOR		0xa0 +/*				0xb0	reserved */ +/*				0xc0	reserved */ +/*				0xd0	reserved */ +/*				0xe0	reserved */ +/*				0xf0	reserved */ + +#define		BPF_JA		0x00 +#define		BPF_JEQ		0x10 +#define		BPF_JGT		0x20 +#define		BPF_JGE		0x30 +#define		BPF_JSET	0x40 +/*				0x50	reserved; used on BSD/OS */ +/*				0x60	reserved */ +/*				0x70	reserved */ +/*				0x80	reserved */ +/*				0x90	reserved */ +/*				0xa0	reserved */ +/*				0xb0	reserved */ +/*				0xc0	reserved */ +/*				0xd0	reserved */ +/*				0xe0	reserved */ +/*				0xf0	reserved */ +#define BPF_SRC(code)	((code) & 0x08) +#define		BPF_K		0x00 +#define		BPF_X		0x08 + +/* ret - BPF_K and BPF_X also apply */ +#define BPF_RVAL(code)	((code) & 0x18) +#define		BPF_A		0x10 +/*				0x18	reserved */ + +/* misc */ +#define BPF_MISCOP(code) ((code) & 0xf8) +#define		BPF_TAX		0x00 +/*				0x08	reserved */ +/*				0x10	reserved */ +/*				0x18	reserved */ +/* #define	BPF_COP		0x20	NetBSD "coprocessor" extensions */ +/*				0x28	reserved */ +/*				0x30	reserved */ +/*				0x38	reserved */ +/* #define	BPF_COPX	0x40	NetBSD "coprocessor" extensions */ +/*					also used on BSD/OS */ +/*				0x48	reserved */ +/*				0x50	reserved */ +/*				0x58	reserved */ +/*				0x60	reserved */ +/*				0x68	reserved */ +/*				0x70	reserved */ +/*				0x78	reserved */ +#define		BPF_TXA		0x80 +/*				0x88	reserved */ +/*				0x90	reserved */ +/*				0x98	reserved */ +/*				0xa0	reserved */ +/*				0xa8	reserved */ +/*				0xb0	reserved */ +/*				0xb8	reserved */ +/*				0xc0	reserved; used on BSD/OS */ +/*				0xc8	reserved */ +/*				0xd0	reserved */ +/*				0xd8	reserved */ +/*				0xe0	reserved */ +/*				0xe8	reserved */ +/*				0xf0	reserved */ +/*				0xf8	reserved */ + +/* + * The instruction data structure. + */ +struct bpf_insn { +	u_short	code; +	u_char 	jt; +	u_char 	jf; +	bpf_u_int32 k; +}; + +/* + * Auxiliary data, for use when interpreting a filter intended for the + * Linux kernel when the kernel rejects the filter (requiring us to + * run it in userland).  It contains VLAN tag information. + */ +struct bpf_aux_data { +	u_short vlan_tag_present; +	u_short vlan_tag; +}; + +/* + * Macros for insn array initializers. + */ +#define BPF_STMT(code, k) { (u_short)(code), 0, 0, k } +#define BPF_JUMP(code, k, jt, jf) { (u_short)(code), jt, jf, k } + +PCAP_API int bpf_validate(const struct bpf_insn *, int); +PCAP_API u_int bpf_filter(const struct bpf_insn *, const u_char *, u_int, u_int); +extern u_int bpf_filter_with_aux_data(const struct bpf_insn *, const u_char *, u_int, u_int, const struct bpf_aux_data *); + +/* + * Number of scratch memory words (for BPF_LD|BPF_MEM and BPF_ST). + */ +#define BPF_MEMWORDS 16 + +#ifdef __cplusplus +} +#endif + +#endif /* !defined(_NET_BPF_H_) && !defined(_BPF_H_) && !defined(_H_BPF) && !defined(lib_pcap_bpf_h) */ diff --git a/src/frontend/qt_sdl/pcap/can_socketcan.h b/src/frontend/qt_sdl/pcap/can_socketcan.h new file mode 100644 index 0000000..332d9ff --- /dev/null +++ b/src/frontend/qt_sdl/pcap/can_socketcan.h @@ -0,0 +1,56 @@ +/*- + * Copyright (c) 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997 + *	The Regents of the University of California.  All rights reserved. + * + * This code is derived from the Stanford/CMU enet packet filter, + * (net/enet.c) distributed as part of 4.3BSD, and code contributed + * to Berkeley by Steven McCanne and Van Jacobson both of Lawrence + * Berkeley Laboratory. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + *    must display the following acknowledgement: + *      This product includes software developed by the University of + *      California, Berkeley and its contributors. + * 4. Neither the name of the University nor the names of its contributors + *    may be used to endorse or promote products derived from this software + *    without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +#ifndef lib_pcap_can_socketcan_h +#define lib_pcap_can_socketcan_h + +#include <pcap/pcap-inttypes.h> + +/* + * SocketCAN header, as per Documentation/networking/can.txt in the + * Linux source. + */ +typedef struct { +	uint32_t can_id; +	uint8_t payload_length; +	uint8_t pad; +	uint8_t reserved1; +	uint8_t reserved2; +} pcap_can_socketcan_hdr; + +#endif diff --git a/src/frontend/qt_sdl/pcap/compiler-tests.h b/src/frontend/qt_sdl/pcap/compiler-tests.h new file mode 100644 index 0000000..8876c67 --- /dev/null +++ b/src/frontend/qt_sdl/pcap/compiler-tests.h @@ -0,0 +1,152 @@ +/* -*- Mode: c; tab-width: 8; indent-tabs-mode: 1; c-basic-offset: 8; -*- */ +/* + * Copyright (c) 1993, 1994, 1995, 1996, 1997 + *	The Regents of the University of California.  All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + *    must display the following acknowledgement: + *	This product includes software developed by the Computer Systems + *	Engineering Group at Lawrence Berkeley Laboratory. + * 4. Neither the name of the University nor of the Laboratory may be used + *    to endorse or promote products derived from this software without + *    specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +#ifndef lib_pcap_compiler_tests_h +#define lib_pcap_compiler_tests_h + + +/* + * This was introduced by Clang: + * + *     http://clang.llvm.org/docs/LanguageExtensions.html#has-attribute + * + * in some version (which version?); it has been picked up by GCC 5.0. + */ +#ifndef __has_attribute +  /* +   * It's a macro, so you can check whether it's defined to check +   * whether it's supported. +   * +   * If it's not, define it to always return 0, so that we move on to +   * the fallback checks. +   */ +  #define __has_attribute(x) 0 +#endif + +/* + * Note that the C90 spec's "6.8.1 Conditional inclusion" and the + * C99 spec's and C11 spec's "6.10.1 Conditional inclusion" say: + * + *    Prior to evaluation, macro invocations in the list of preprocessing + *    tokens that will become the controlling constant expression are + *    replaced (except for those macro names modified by the defined unary + *    operator), just as in normal text.  If the token "defined" is + *    generated as a result of this replacement process or use of the + *    "defined" unary operator does not match one of the two specified + *    forms prior to macro replacement, the behavior is undefined. + * + * so you shouldn't use defined() in a #define that's used in #if or + * #elif.  Some versions of Clang, for example, will warn about this. + * + * Instead, we check whether the pre-defined macros for particular + * compilers are defined and, if not, define the "is this version XXX + * or a later version of this compiler" macros as 0. + */ + +/* + * Check whether this is GCC major.minor or a later release, or some + * compiler that claims to be "just like GCC" of that version or a + * later release. + */ + +#if ! defined(__GNUC__) +#define PCAP_IS_AT_LEAST_GNUC_VERSION(major, minor) 0 +#else +#define PCAP_IS_AT_LEAST_GNUC_VERSION(major, minor) \ +	(__GNUC__ > (major) || \ +	 (__GNUC__ == (major) && __GNUC_MINOR__ >= (minor))) +#endif + +/* + * Check whether this is Sun C/SunPro C/Oracle Studio major.minor + * or a later release. + * + * The version number in __SUNPRO_C is encoded in hex BCD, with the + * uppermost hex digit being the major version number, the next + * one or two hex digits being the minor version number, and + * the last digit being the patch version. + * + * It represents the *compiler* version, not the product version; + * see + * + *    https://sourceforge.net/p/predef/wiki/Compilers/ + * + * for a partial mapping, which we assume continues for later + * 12.x product releases. + */ + +#if ! defined(__SUNPRO_C) +#define PCAP_IS_AT_LEAST_SUNC_VERSION(major,minor) 0 +#else +#define PCAP_SUNPRO_VERSION_TO_BCD(major, minor) \ +	(((minor) >= 10) ? \ +	    (((major) << 12) | (((minor)/10) << 8) | (((minor)%10) << 4)) : \ +	    (((major) << 8) | ((minor) << 4))) +#define PCAP_IS_AT_LEAST_SUNC_VERSION(major,minor) \ +	(__SUNPRO_C >= PCAP_SUNPRO_VERSION_TO_BCD((major), (minor))) +#endif + +/* + * Check whether this is IBM XL C major.minor or a later release. + * + * The version number in __xlC__ has the major version in the + * upper 8 bits and the minor version in the lower 8 bits. + */ + +#if ! defined(__xlC__) +#define PCAP_IS_AT_LEAST_XL_C_VERSION(major,minor) 0 +#else +#define PCAP_IS_AT_LEAST_XL_C_VERSION(major, minor) \ +	(__xlC__ >= (((major) << 8) | (minor))) +#endif + +/* + * Check whether this is HP aC++/HP C major.minor or a later release. + * + * The version number in __HP_aCC is encoded in zero-padded decimal BCD, + * with the "A." stripped off, the uppermost two decimal digits being + * the major version number, the next two decimal digits being the minor + * version number, and the last two decimal digits being the patch version. + * (Strip off the A., remove the . between the major and minor version + * number, and add two digits of patch.) + */ + +#if ! defined(__HP_aCC) +#define PCAP_IS_AT_LEAST_HP_C_VERSION(major,minor) 0 +#else +#define PCAP_IS_AT_LEAST_HP_C_VERSION(major,minor) \ +	(__HP_aCC >= ((major)*10000 + (minor)*100)) +#endif + +#endif /* lib_pcap_funcattrs_h */ diff --git a/src/frontend/qt_sdl/pcap/dlt.h b/src/frontend/qt_sdl/pcap/dlt.h new file mode 100644 index 0000000..609bcaf --- /dev/null +++ b/src/frontend/qt_sdl/pcap/dlt.h @@ -0,0 +1,1389 @@ +/*- + * Copyright (c) 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997 + *	The Regents of the University of California.  All rights reserved. + * + * This code is derived from the Stanford/CMU enet packet filter, + * (net/enet.c) distributed as part of 4.3BSD, and code contributed + * to Berkeley by Steven McCanne and Van Jacobson both of Lawrence + * Berkeley Laboratory. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. Neither the name of the University nor the names of its contributors + *    may be used to endorse or promote products derived from this software + *    without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + * + *      @(#)bpf.h       7.1 (Berkeley) 5/7/91 + */ + +#ifndef lib_pcap_dlt_h +#define lib_pcap_dlt_h + +/* + * Link-layer header type codes. + * + * Do *NOT* add new values to this list without asking + * "tcpdump-workers@lists.tcpdump.org" for a value.  Otherwise, you run + * the risk of using a value that's already being used for some other + * purpose, and of having tools that read libpcap-format captures not + * being able to handle captures with your new DLT_ value, with no hope + * that they will ever be changed to do so (as that would destroy their + * ability to read captures using that value for that other purpose). + * + * See + * + *	http://www.tcpdump.org/linktypes.html + * + * for detailed descriptions of some of these link-layer header types. + */ + +/* + * These are the types that are the same on all platforms, and that + * have been defined by <net/bpf.h> for ages. + */ +#define DLT_NULL	0	/* BSD loopback encapsulation */ +#define DLT_EN10MB	1	/* Ethernet (10Mb) */ +#define DLT_EN3MB	2	/* Experimental Ethernet (3Mb) */ +#define DLT_AX25	3	/* Amateur Radio AX.25 */ +#define DLT_PRONET	4	/* Proteon ProNET Token Ring */ +#define DLT_CHAOS	5	/* Chaos */ +#define DLT_IEEE802	6	/* 802.5 Token Ring */ +#define DLT_ARCNET	7	/* ARCNET, with BSD-style header */ +#define DLT_SLIP	8	/* Serial Line IP */ +#define DLT_PPP		9	/* Point-to-point Protocol */ +#define DLT_FDDI	10	/* FDDI */ + +/* + * These are types that are different on some platforms, and that + * have been defined by <net/bpf.h> for ages.  We use #ifdefs to + * detect the BSDs that define them differently from the traditional + * libpcap <net/bpf.h> + * + * XXX - DLT_ATM_RFC1483 is 13 in BSD/OS, and DLT_RAW is 14 in BSD/OS, + * but I don't know what the right #define is for BSD/OS. + */ +#define DLT_ATM_RFC1483	11	/* LLC-encapsulated ATM */ + +#ifdef __OpenBSD__ +#define DLT_RAW		14	/* raw IP */ +#else +#define DLT_RAW		12	/* raw IP */ +#endif + +/* + * Given that the only OS that currently generates BSD/OS SLIP or PPP + * is, well, BSD/OS, arguably everybody should have chosen its values + * for DLT_SLIP_BSDOS and DLT_PPP_BSDOS, which are 15 and 16, but they + * didn't.  So it goes. + */ +#if defined(__NetBSD__) || defined(__FreeBSD__) +#ifndef DLT_SLIP_BSDOS +#define DLT_SLIP_BSDOS	13	/* BSD/OS Serial Line IP */ +#define DLT_PPP_BSDOS	14	/* BSD/OS Point-to-point Protocol */ +#endif +#else +#define DLT_SLIP_BSDOS	15	/* BSD/OS Serial Line IP */ +#define DLT_PPP_BSDOS	16	/* BSD/OS Point-to-point Protocol */ +#endif + +/* + * 17 was used for DLT_PFLOG in OpenBSD; it no longer is. + * + * It was DLT_LANE8023 in SuSE 6.3, so we defined LINKTYPE_PFLOG + * as 117 so that pflog captures would use a link-layer header type + * value that didn't collide with any other values.  On all + * platforms other than OpenBSD, we defined DLT_PFLOG as 117, + * and we mapped between LINKTYPE_PFLOG and DLT_PFLOG. + * + * OpenBSD eventually switched to using 117 for DLT_PFLOG as well. + * + * Don't use 17 for anything else. + */ + +/* + * 18 is used for DLT_PFSYNC in OpenBSD, NetBSD, DragonFly BSD and + * Mac OS X; don't use it for anything else.  (FreeBSD uses 121, + * which collides with DLT_HHDLC, even though it doesn't use 18 + * for anything and doesn't appear to have ever used it for anything.) + * + * We define it as 18 on those platforms; it is, unfortunately, used + * for DLT_CIP in Suse 6.3, so we don't define it as DLT_PFSYNC + * in general.  As the packet format for it, like that for + * DLT_PFLOG, is not only OS-dependent but OS-version-dependent, + * we don't support printing it in tcpdump except on OSes that + * have the relevant header files, so it's not that useful on + * other platforms. + */ +#if defined(__OpenBSD__) || defined(__NetBSD__) || defined(__DragonFly__) || defined(__APPLE__) +#define DLT_PFSYNC	18 +#endif + +#define DLT_ATM_CLIP	19	/* Linux Classical-IP over ATM */ + +/* + * Apparently Redback uses this for its SmartEdge 400/800.  I hope + * nobody else decided to use it, too. + */ +#define DLT_REDBACK_SMARTEDGE	32 + +/* + * These values are defined by NetBSD; other platforms should refrain from + * using them for other purposes, so that NetBSD savefiles with link + * types of 50 or 51 can be read as this type on all platforms. + */ +#define DLT_PPP_SERIAL	50	/* PPP over serial with HDLC encapsulation */ +#define DLT_PPP_ETHER	51	/* PPP over Ethernet */ + +/* + * The Axent Raptor firewall - now the Symantec Enterprise Firewall - uses + * a link-layer type of 99 for the tcpdump it supplies.  The link-layer + * header has 6 bytes of unknown data, something that appears to be an + * Ethernet type, and 36 bytes that appear to be 0 in at least one capture + * I've seen. + */ +#define DLT_SYMANTEC_FIREWALL	99 + +/* + * Values between 100 and 103 are used in capture file headers as + * link-layer header type LINKTYPE_ values corresponding to DLT_ types + * that differ between platforms; don't use those values for new DLT_ + * new types. + */ + +/* + * Values starting with 104 are used for newly-assigned link-layer + * header type values; for those link-layer header types, the DLT_ + * value returned by pcap_datalink() and passed to pcap_open_dead(), + * and the LINKTYPE_ value that appears in capture files, are the + * same. + * + * DLT_MATCHING_MIN is the lowest such value; DLT_MATCHING_MAX is + * the highest such value. + */ +#define DLT_MATCHING_MIN	104 + +/* + * This value was defined by libpcap 0.5; platforms that have defined + * it with a different value should define it here with that value - + * a link type of 104 in a save file will be mapped to DLT_C_HDLC, + * whatever value that happens to be, so programs will correctly + * handle files with that link type regardless of the value of + * DLT_C_HDLC. + * + * The name DLT_C_HDLC was used by BSD/OS; we use that name for source + * compatibility with programs written for BSD/OS. + * + * libpcap 0.5 defined it as DLT_CHDLC; we define DLT_CHDLC as well, + * for source compatibility with programs written for libpcap 0.5. + */ +#define DLT_C_HDLC	104	/* Cisco HDLC */ +#define DLT_CHDLC	DLT_C_HDLC + +#define DLT_IEEE802_11	105	/* IEEE 802.11 wireless */ + +/* + * 106 is reserved for Linux Classical IP over ATM; it's like DLT_RAW, + * except when it isn't.  (I.e., sometimes it's just raw IP, and + * sometimes it isn't.)  We currently handle it as DLT_LINUX_SLL, + * so that we don't have to worry about the link-layer header.) + */ + +/* + * Frame Relay; BSD/OS has a DLT_FR with a value of 11, but that collides + * with other values. + * DLT_FR and DLT_FRELAY packets start with the Q.922 Frame Relay header + * (DLCI, etc.). + */ +#define DLT_FRELAY	107 + +/* + * OpenBSD DLT_LOOP, for loopback devices; it's like DLT_NULL, except + * that the AF_ type in the link-layer header is in network byte order. + * + * DLT_LOOP is 12 in OpenBSD, but that's DLT_RAW in other OSes, so + * we don't use 12 for it in OSes other than OpenBSD. + */ +#ifdef __OpenBSD__ +#define DLT_LOOP	12 +#else +#define DLT_LOOP	108 +#endif + +/* + * Encapsulated packets for IPsec; DLT_ENC is 13 in OpenBSD, but that's + * DLT_SLIP_BSDOS in NetBSD, so we don't use 13 for it in OSes other + * than OpenBSD. + */ +#ifdef __OpenBSD__ +#define DLT_ENC		13 +#else +#define DLT_ENC		109 +#endif + +/* + * Values between 110 and 112 are reserved for use in capture file headers + * as link-layer types corresponding to DLT_ types that might differ + * between platforms; don't use those values for new DLT_ types + * other than the corresponding DLT_ types. + */ + +/* + * This is for Linux cooked sockets. + */ +#define DLT_LINUX_SLL	113 + +/* + * Apple LocalTalk hardware. + */ +#define DLT_LTALK	114 + +/* + * Acorn Econet. + */ +#define DLT_ECONET	115 + +/* + * Reserved for use with OpenBSD ipfilter. + */ +#define DLT_IPFILTER	116 + +/* + * OpenBSD DLT_PFLOG. + */ +#define DLT_PFLOG	117 + +/* + * Registered for Cisco-internal use. + */ +#define DLT_CISCO_IOS	118 + +/* + * For 802.11 cards using the Prism II chips, with a link-layer + * header including Prism monitor mode information plus an 802.11 + * header. + */ +#define DLT_PRISM_HEADER	119 + +/* + * Reserved for Aironet 802.11 cards, with an Aironet link-layer header + * (see Doug Ambrisko's FreeBSD patches). + */ +#define DLT_AIRONET_HEADER	120 + +/* + * Sigh. + * + * 121 was reserved for Siemens HiPath HDLC on 2002-01-25, as + * requested by Tomas Kukosa. + * + * On 2004-02-25, a FreeBSD checkin to sys/net/bpf.h was made that + * assigned 121 as DLT_PFSYNC.  In current versions, its libpcap + * does DLT_ <-> LINKTYPE_ mapping, mapping DLT_PFSYNC to a + * LINKTYPE_PFSYNC value of 246, so it should write out DLT_PFSYNC + * dump files with 246 as the link-layer header type.  (Earlier + * versions might not have done mapping, in which case they would + * have written them out with a link-layer header type of 121.) + * + * OpenBSD, from which pf came, however, uses 18 for DLT_PFSYNC; + * its libpcap does no DLT_ <-> LINKTYPE_ mapping, so it would + * write out DLT_PFSYNC dump files with use 18 as the link-layer + * header type. + * + * NetBSD, DragonFly BSD, and Darwin also use 18 for DLT_PFSYNC; in + * current versions, their libpcaps do DLT_ <-> LINKTYPE_ mapping, + * mapping DLT_PFSYNC to a LINKTYPE_PFSYNC value of 246, so they + * should write out DLT_PFSYNC dump files with 246 as the link-layer + * header type.  (Earlier versions might not have done mapping, + * in which case they'd work the same way OpenBSD does, writing + * them out with a link-layer header type of 18.) + * + * We'll define DLT_PFSYNC as: + * + *    18 on NetBSD, OpenBSD, DragonFly BSD, and Darwin; + * + *    121 on FreeBSD; + * + *    246 everywhere else. + * + * We'll define DLT_HHDLC as 121 on everything except for FreeBSD; + * anybody who wants to compile, on FreeBSD, code that uses DLT_HHDLC + * is out of luck. + * + * We'll define LINKTYPE_PFSYNC as 246 on *all* platforms, so that + * savefiles written using *this* code won't use 18 or 121 for PFSYNC, + * they'll all use 246. + * + * Code that uses pcap_datalink() to determine the link-layer header + * type of a savefile won't, when built and run on FreeBSD, be able + * to distinguish between LINKTYPE_PFSYNC and LINKTYPE_HHDLC capture + * files, as pcap_datalink() will give 121 for both of them.  Code + * that doesn't, such as the code in Wireshark, will be able to + * distinguish between them. + * + * FreeBSD's libpcap won't map a link-layer header type of 18 - i.e., + * DLT_PFSYNC files from OpenBSD and possibly older versions of NetBSD, + * DragonFly BSD, and OS X - to DLT_PFSYNC, so code built with FreeBSD's + * libpcap won't treat those files as DLT_PFSYNC files. + * + * Other libpcaps won't map a link-layer header type of 121 to DLT_PFSYNC; + * this means they can read DLT_HHDLC files, if any exist, but won't + * treat pcap files written by any older versions of FreeBSD libpcap that + * didn't map to 246 as DLT_PFSYNC files. + */ +#ifdef __FreeBSD__ +#define DLT_PFSYNC		121 +#else +#define DLT_HHDLC		121 +#endif + +/* + * This is for RFC 2625 IP-over-Fibre Channel. + * + * This is not for use with raw Fibre Channel, where the link-layer + * header starts with a Fibre Channel frame header; it's for IP-over-FC, + * where the link-layer header starts with an RFC 2625 Network_Header + * field. + */ +#define DLT_IP_OVER_FC		122 + +/* + * This is for Full Frontal ATM on Solaris with SunATM, with a + * pseudo-header followed by an AALn PDU. + * + * There may be other forms of Full Frontal ATM on other OSes, + * with different pseudo-headers. + * + * If ATM software returns a pseudo-header with VPI/VCI information + * (and, ideally, packet type information, e.g. signalling, ILMI, + * LANE, LLC-multiplexed traffic, etc.), it should not use + * DLT_ATM_RFC1483, but should get a new DLT_ value, so tcpdump + * and the like don't have to infer the presence or absence of a + * pseudo-header and the form of the pseudo-header. + */ +#define DLT_SUNATM		123	/* Solaris+SunATM */ + +/* + * Reserved as per request from Kent Dahlgren <kent@praesum.com> + * for private use. + */ +#define DLT_RIO                 124     /* RapidIO */ +#define DLT_PCI_EXP             125     /* PCI Express */ +#define DLT_AURORA              126     /* Xilinx Aurora link layer */ + +/* + * Header for 802.11 plus a number of bits of link-layer information + * including radio information, used by some recent BSD drivers as + * well as the madwifi Atheros driver for Linux. + */ +#define DLT_IEEE802_11_RADIO	127	/* 802.11 plus radiotap radio header */ + +/* + * Reserved for the TZSP encapsulation, as per request from + * Chris Waters <chris.waters@networkchemistry.com> + * TZSP is a generic encapsulation for any other link type, + * which includes a means to include meta-information + * with the packet, e.g. signal strength and channel + * for 802.11 packets. + */ +#define DLT_TZSP                128     /* Tazmen Sniffer Protocol */ + +/* + * BSD's ARCNET headers have the source host, destination host, + * and type at the beginning of the packet; that's what's handed + * up to userland via BPF. + * + * Linux's ARCNET headers, however, have a 2-byte offset field + * between the host IDs and the type; that's what's handed up + * to userland via PF_PACKET sockets. + * + * We therefore have to have separate DLT_ values for them. + */ +#define DLT_ARCNET_LINUX	129	/* ARCNET */ + +/* + * Juniper-private data link types, as per request from + * Hannes Gredler <hannes@juniper.net>.  The DLT_s are used + * for passing on chassis-internal metainformation such as + * QOS profiles, etc.. + */ +#define DLT_JUNIPER_MLPPP       130 +#define DLT_JUNIPER_MLFR        131 +#define DLT_JUNIPER_ES          132 +#define DLT_JUNIPER_GGSN        133 +#define DLT_JUNIPER_MFR         134 +#define DLT_JUNIPER_ATM2        135 +#define DLT_JUNIPER_SERVICES    136 +#define DLT_JUNIPER_ATM1        137 + +/* + * Apple IP-over-IEEE 1394, as per a request from Dieter Siegmund + * <dieter@apple.com>.  The header that's presented is an Ethernet-like + * header: + * + *	#define FIREWIRE_EUI64_LEN	8 + *	struct firewire_header { + *		u_char  firewire_dhost[FIREWIRE_EUI64_LEN]; + *		u_char  firewire_shost[FIREWIRE_EUI64_LEN]; + *		u_short firewire_type; + *	}; + * + * with "firewire_type" being an Ethernet type value, rather than, + * for example, raw GASP frames being handed up. + */ +#define DLT_APPLE_IP_OVER_IEEE1394	138 + +/* + * Various SS7 encapsulations, as per a request from Jeff Morriss + * <jeff.morriss[AT]ulticom.com> and subsequent discussions. + */ +#define DLT_MTP2_WITH_PHDR	139	/* pseudo-header with various info, followed by MTP2 */ +#define DLT_MTP2		140	/* MTP2, without pseudo-header */ +#define DLT_MTP3		141	/* MTP3, without pseudo-header or MTP2 */ +#define DLT_SCCP		142	/* SCCP, without pseudo-header or MTP2 or MTP3 */ + +/* + * DOCSIS MAC frames. + */ +#define DLT_DOCSIS		143 + +/* + * Linux-IrDA packets. Protocol defined at http://www.irda.org. + * Those packets include IrLAP headers and above (IrLMP...), but + * don't include Phy framing (SOF/EOF/CRC & byte stuffing), because Phy + * framing can be handled by the hardware and depend on the bitrate. + * This is exactly the format you would get capturing on a Linux-IrDA + * interface (irdaX), but not on a raw serial port. + * Note the capture is done in "Linux-cooked" mode, so each packet include + * a fake packet header (struct sll_header). This is because IrDA packet + * decoding is dependant on the direction of the packet (incomming or + * outgoing). + * When/if other platform implement IrDA capture, we may revisit the + * issue and define a real DLT_IRDA... + * Jean II + */ +#define DLT_LINUX_IRDA		144 + +/* + * Reserved for IBM SP switch and IBM Next Federation switch. + */ +#define DLT_IBM_SP		145 +#define DLT_IBM_SN		146 + +/* + * Reserved for private use.  If you have some link-layer header type + * that you want to use within your organization, with the capture files + * using that link-layer header type not ever be sent outside your + * organization, you can use these values. + * + * No libpcap release will use these for any purpose, nor will any + * tcpdump release use them, either. + * + * Do *NOT* use these in capture files that you expect anybody not using + * your private versions of capture-file-reading tools to read; in + * particular, do *NOT* use them in products, otherwise you may find that + * people won't be able to use tcpdump, or snort, or Ethereal, or... to + * read capture files from your firewall/intrusion detection/traffic + * monitoring/etc. appliance, or whatever product uses that DLT_ value, + * and you may also find that the developers of those applications will + * not accept patches to let them read those files. + * + * Also, do not use them if somebody might send you a capture using them + * for *their* private type and tools using them for *your* private type + * would have to read them. + * + * Instead, ask "tcpdump-workers@lists.tcpdump.org" for a new DLT_ value, + * as per the comment above, and use the type you're given. + */ +#define DLT_USER0		147 +#define DLT_USER1		148 +#define DLT_USER2		149 +#define DLT_USER3		150 +#define DLT_USER4		151 +#define DLT_USER5		152 +#define DLT_USER6		153 +#define DLT_USER7		154 +#define DLT_USER8		155 +#define DLT_USER9		156 +#define DLT_USER10		157 +#define DLT_USER11		158 +#define DLT_USER12		159 +#define DLT_USER13		160 +#define DLT_USER14		161 +#define DLT_USER15		162 + +/* + * For future use with 802.11 captures - defined by AbsoluteValue + * Systems to store a number of bits of link-layer information + * including radio information: + * + *	http://www.shaftnet.org/~pizza/software/capturefrm.txt + * + * but it might be used by some non-AVS drivers now or in the + * future. + */ +#define DLT_IEEE802_11_RADIO_AVS 163	/* 802.11 plus AVS radio header */ + +/* + * Juniper-private data link type, as per request from + * Hannes Gredler <hannes@juniper.net>.  The DLT_s are used + * for passing on chassis-internal metainformation such as + * QOS profiles, etc.. + */ +#define DLT_JUNIPER_MONITOR     164 + +/* + * BACnet MS/TP frames. + */ +#define DLT_BACNET_MS_TP	165 + +/* + * Another PPP variant as per request from Karsten Keil <kkeil@suse.de>. + * + * This is used in some OSes to allow a kernel socket filter to distinguish + * between incoming and outgoing packets, on a socket intended to + * supply pppd with outgoing packets so it can do dial-on-demand and + * hangup-on-lack-of-demand; incoming packets are filtered out so they + * don't cause pppd to hold the connection up (you don't want random + * input packets such as port scans, packets from old lost connections, + * etc. to force the connection to stay up). + * + * The first byte of the PPP header (0xff03) is modified to accomodate + * the direction - 0x00 = IN, 0x01 = OUT. + */ +#define DLT_PPP_PPPD		166 + +/* + * Names for backwards compatibility with older versions of some PPP + * software; new software should use DLT_PPP_PPPD. + */ +#define DLT_PPP_WITH_DIRECTION	DLT_PPP_PPPD +#define DLT_LINUX_PPP_WITHDIRECTION	DLT_PPP_PPPD + +/* + * Juniper-private data link type, as per request from + * Hannes Gredler <hannes@juniper.net>.  The DLT_s are used + * for passing on chassis-internal metainformation such as + * QOS profiles, cookies, etc.. + */ +#define DLT_JUNIPER_PPPOE       167 +#define DLT_JUNIPER_PPPOE_ATM   168 + +#define DLT_GPRS_LLC		169	/* GPRS LLC */ +#define DLT_GPF_T		170	/* GPF-T (ITU-T G.7041/Y.1303) */ +#define DLT_GPF_F		171	/* GPF-F (ITU-T G.7041/Y.1303) */ + +/* + * Requested by Oolan Zimmer <oz@gcom.com> for use in Gcom's T1/E1 line + * monitoring equipment. + */ +#define DLT_GCOM_T1E1		172 +#define DLT_GCOM_SERIAL		173 + +/* + * Juniper-private data link type, as per request from + * Hannes Gredler <hannes@juniper.net>.  The DLT_ is used + * for internal communication to Physical Interface Cards (PIC) + */ +#define DLT_JUNIPER_PIC_PEER    174 + +/* + * Link types requested by Gregor Maier <gregor@endace.com> of Endace + * Measurement Systems.  They add an ERF header (see + * http://www.endace.com/support/EndaceRecordFormat.pdf) in front of + * the link-layer header. + */ +#define DLT_ERF_ETH		175	/* Ethernet */ +#define DLT_ERF_POS		176	/* Packet-over-SONET */ + +/* + * Requested by Daniele Orlandi <daniele@orlandi.com> for raw LAPD + * for vISDN (http://www.orlandi.com/visdn/).  Its link-layer header + * includes additional information before the LAPD header, so it's + * not necessarily a generic LAPD header. + */ +#define DLT_LINUX_LAPD		177 + +/* + * Juniper-private data link type, as per request from + * Hannes Gredler <hannes@juniper.net>. + * The DLT_ are used for prepending meta-information + * like interface index, interface name + * before standard Ethernet, PPP, Frelay & C-HDLC Frames + */ +#define DLT_JUNIPER_ETHER       178 +#define DLT_JUNIPER_PPP         179 +#define DLT_JUNIPER_FRELAY      180 +#define DLT_JUNIPER_CHDLC       181 + +/* + * Multi Link Frame Relay (FRF.16) + */ +#define DLT_MFR                 182 + +/* + * Juniper-private data link type, as per request from + * Hannes Gredler <hannes@juniper.net>. + * The DLT_ is used for internal communication with a + * voice Adapter Card (PIC) + */ +#define DLT_JUNIPER_VP          183 + +/* + * Arinc 429 frames. + * DLT_ requested by Gianluca Varenni <gianluca.varenni@cacetech.com>. + * Every frame contains a 32bit A429 label. + * More documentation on Arinc 429 can be found at + * http://www.condoreng.com/support/downloads/tutorials/ARINCTutorial.pdf + */ +#define DLT_A429                184 + +/* + * Arinc 653 Interpartition Communication messages. + * DLT_ requested by Gianluca Varenni <gianluca.varenni@cacetech.com>. + * Please refer to the A653-1 standard for more information. + */ +#define DLT_A653_ICM            185 + +/* + * This used to be "USB packets, beginning with a USB setup header; + * requested by Paolo Abeni <paolo.abeni@email.it>." + * + * However, that header didn't work all that well - it left out some + * useful information - and was abandoned in favor of the DLT_USB_LINUX + * header. + * + * This is now used by FreeBSD for its BPF taps for USB; that has its + * own headers.  So it is written, so it is done. + * + * For source-code compatibility, we also define DLT_USB to have this + * value.  We do it numerically so that, if code that includes this + * file (directly or indirectly) also includes an OS header that also + * defines DLT_USB as 186, we don't get a redefinition warning. + * (NetBSD 7 does that.) + */ +#define DLT_USB_FREEBSD		186 +#define DLT_USB			186 + +/* + * Bluetooth HCI UART transport layer (part H:4); requested by + * Paolo Abeni. + */ +#define DLT_BLUETOOTH_HCI_H4	187 + +/* + * IEEE 802.16 MAC Common Part Sublayer; requested by Maria Cruz + * <cruz_petagay@bah.com>. + */ +#define DLT_IEEE802_16_MAC_CPS	188 + +/* + * USB packets, beginning with a Linux USB header; requested by + * Paolo Abeni <paolo.abeni@email.it>. + */ +#define DLT_USB_LINUX		189 + +/* + * Controller Area Network (CAN) v. 2.0B packets. + * DLT_ requested by Gianluca Varenni <gianluca.varenni@cacetech.com>. + * Used to dump CAN packets coming from a CAN Vector board. + * More documentation on the CAN v2.0B frames can be found at + * http://www.can-cia.org/downloads/?269 + */ +#define DLT_CAN20B              190 + +/* + * IEEE 802.15.4, with address fields padded, as is done by Linux + * drivers; requested by Juergen Schimmer. + */ +#define DLT_IEEE802_15_4_LINUX	191 + +/* + * Per Packet Information encapsulated packets. + * DLT_ requested by Gianluca Varenni <gianluca.varenni@cacetech.com>. + */ +#define DLT_PPI			192 + +/* + * Header for 802.16 MAC Common Part Sublayer plus a radiotap radio header; + * requested by Charles Clancy. + */ +#define DLT_IEEE802_16_MAC_CPS_RADIO	193 + +/* + * Juniper-private data link type, as per request from + * Hannes Gredler <hannes@juniper.net>. + * The DLT_ is used for internal communication with a + * integrated service module (ISM). + */ +#define DLT_JUNIPER_ISM         194 + +/* + * IEEE 802.15.4, exactly as it appears in the spec (no padding, no + * nothing); requested by Mikko Saarnivala <mikko.saarnivala@sensinode.com>. + * For this one, we expect the FCS to be present at the end of the frame; + * if the frame has no FCS, DLT_IEEE802_15_4_NOFCS should be used. + */ +#define DLT_IEEE802_15_4	195 + +/* + * Various link-layer types, with a pseudo-header, for SITA + * (http://www.sita.aero/); requested by Fulko Hew (fulko.hew@gmail.com). + */ +#define DLT_SITA		196 + +/* + * Various link-layer types, with a pseudo-header, for Endace DAG cards; + * encapsulates Endace ERF records.  Requested by Stephen Donnelly + * <stephen@endace.com>. + */ +#define DLT_ERF			197 + +/* + * Special header prepended to Ethernet packets when capturing from a + * u10 Networks board.  Requested by Phil Mulholland + * <phil@u10networks.com>. + */ +#define DLT_RAIF1		198 + +/* + * IPMB packet for IPMI, beginning with the I2C slave address, followed + * by the netFn and LUN, etc..  Requested by Chanthy Toeung + * <chanthy.toeung@ca.kontron.com>. + */ +#define DLT_IPMB		199 + +/* + * Juniper-private data link type, as per request from + * Hannes Gredler <hannes@juniper.net>. + * The DLT_ is used for capturing data on a secure tunnel interface. + */ +#define DLT_JUNIPER_ST          200 + +/* + * Bluetooth HCI UART transport layer (part H:4), with pseudo-header + * that includes direction information; requested by Paolo Abeni. + */ +#define DLT_BLUETOOTH_HCI_H4_WITH_PHDR	201 + +/* + * AX.25 packet with a 1-byte KISS header; see + * + *	http://www.ax25.net/kiss.htm + * + * as per Richard Stearn <richard@rns-stearn.demon.co.uk>. + */ +#define DLT_AX25_KISS		202 + +/* + * LAPD packets from an ISDN channel, starting with the address field, + * with no pseudo-header. + * Requested by Varuna De Silva <varunax@gmail.com>. + */ +#define DLT_LAPD		203 + +/* + * Variants of various link-layer headers, with a one-byte direction + * pseudo-header prepended - zero means "received by this host", + * non-zero (any non-zero value) means "sent by this host" - as per + * Will Barker <w.barker@zen.co.uk>. + */ +#define DLT_PPP_WITH_DIR	204	/* PPP - don't confuse with DLT_PPP_WITH_DIRECTION */ +#define DLT_C_HDLC_WITH_DIR	205	/* Cisco HDLC */ +#define DLT_FRELAY_WITH_DIR	206	/* Frame Relay */ +#define DLT_LAPB_WITH_DIR	207	/* LAPB */ + +/* + * 208 is reserved for an as-yet-unspecified proprietary link-layer + * type, as requested by Will Barker. + */ + +/* + * IPMB with a Linux-specific pseudo-header; as requested by Alexey Neyman + * <avn@pigeonpoint.com>. + */ +#define DLT_IPMB_LINUX		209 + +/* + * FlexRay automotive bus - http://www.flexray.com/ - as requested + * by Hannes Kaelber <hannes.kaelber@x2e.de>. + */ +#define DLT_FLEXRAY		210 + +/* + * Media Oriented Systems Transport (MOST) bus for multimedia + * transport - http://www.mostcooperation.com/ - as requested + * by Hannes Kaelber <hannes.kaelber@x2e.de>. + */ +#define DLT_MOST		211 + +/* + * Local Interconnect Network (LIN) bus for vehicle networks - + * http://www.lin-subbus.org/ - as requested by Hannes Kaelber + * <hannes.kaelber@x2e.de>. + */ +#define DLT_LIN			212 + +/* + * X2E-private data link type used for serial line capture, + * as requested by Hannes Kaelber <hannes.kaelber@x2e.de>. + */ +#define DLT_X2E_SERIAL		213 + +/* + * X2E-private data link type used for the Xoraya data logger + * family, as requested by Hannes Kaelber <hannes.kaelber@x2e.de>. + */ +#define DLT_X2E_XORAYA		214 + +/* + * IEEE 802.15.4, exactly as it appears in the spec (no padding, no + * nothing), but with the PHY-level data for non-ASK PHYs (4 octets + * of 0 as preamble, one octet of SFD, one octet of frame length+ + * reserved bit, and then the MAC-layer data, starting with the + * frame control field). + * + * Requested by Max Filippov <jcmvbkbc@gmail.com>. + */ +#define DLT_IEEE802_15_4_NONASK_PHY	215 + +/* + * David Gibson <david@gibson.dropbear.id.au> requested this for + * captures from the Linux kernel /dev/input/eventN devices. This + * is used to communicate keystrokes and mouse movements from the + * Linux kernel to display systems, such as Xorg. + */ +#define DLT_LINUX_EVDEV		216 + +/* + * GSM Um and Abis interfaces, preceded by a "gsmtap" header. + * + * Requested by Harald Welte <laforge@gnumonks.org>. + */ +#define DLT_GSMTAP_UM		217 +#define DLT_GSMTAP_ABIS		218 + +/* + * MPLS, with an MPLS label as the link-layer header. + * Requested by Michele Marchetto <michele@openbsd.org> on behalf + * of OpenBSD. + */ +#define DLT_MPLS		219 + +/* + * USB packets, beginning with a Linux USB header, with the USB header + * padded to 64 bytes; required for memory-mapped access. + */ +#define DLT_USB_LINUX_MMAPPED	220 + +/* + * DECT packets, with a pseudo-header; requested by + * Matthias Wenzel <tcpdump@mazzoo.de>. + */ +#define DLT_DECT		221 + +/* + * From: "Lidwa, Eric (GSFC-582.0)[SGT INC]" <eric.lidwa-1@nasa.gov> + * Date: Mon, 11 May 2009 11:18:30 -0500 + * + * DLT_AOS. We need it for AOS Space Data Link Protocol. + *   I have already written dissectors for but need an OK from + *   legal before I can submit a patch. + * + */ +#define DLT_AOS                 222 + +/* + * Wireless HART (Highway Addressable Remote Transducer) + * From the HART Communication Foundation + * IES/PAS 62591 + * + * Requested by Sam Roberts <vieuxtech@gmail.com>. + */ +#define DLT_WIHART		223 + +/* + * Fibre Channel FC-2 frames, beginning with a Frame_Header. + * Requested by Kahou Lei <kahou82@gmail.com>. + */ +#define DLT_FC_2		224 + +/* + * Fibre Channel FC-2 frames, beginning with an encoding of the + * SOF, and ending with an encoding of the EOF. + * + * The encodings represent the frame delimiters as 4-byte sequences + * representing the corresponding ordered sets, with K28.5 + * represented as 0xBC, and the D symbols as the corresponding + * byte values; for example, SOFi2, which is K28.5 - D21.5 - D1.2 - D21.2, + * is represented as 0xBC 0xB5 0x55 0x55. + * + * Requested by Kahou Lei <kahou82@gmail.com>. + */ +#define DLT_FC_2_WITH_FRAME_DELIMS	225 + +/* + * Solaris ipnet pseudo-header; requested by Darren Reed <Darren.Reed@Sun.COM>. + * + * The pseudo-header starts with a one-byte version number; for version 2, + * the pseudo-header is: + * + * struct dl_ipnetinfo { + *     uint8_t   dli_version; + *     uint8_t   dli_family; + *     uint16_t  dli_htype; + *     uint32_t  dli_pktlen; + *     uint32_t  dli_ifindex; + *     uint32_t  dli_grifindex; + *     uint32_t  dli_zsrc; + *     uint32_t  dli_zdst; + * }; + * + * dli_version is 2 for the current version of the pseudo-header. + * + * dli_family is a Solaris address family value, so it's 2 for IPv4 + * and 26 for IPv6. + * + * dli_htype is a "hook type" - 0 for incoming packets, 1 for outgoing + * packets, and 2 for packets arriving from another zone on the same + * machine. + * + * dli_pktlen is the length of the packet data following the pseudo-header + * (so the captured length minus dli_pktlen is the length of the + * pseudo-header, assuming the entire pseudo-header was captured). + * + * dli_ifindex is the interface index of the interface on which the + * packet arrived. + * + * dli_grifindex is the group interface index number (for IPMP interfaces). + * + * dli_zsrc is the zone identifier for the source of the packet. + * + * dli_zdst is the zone identifier for the destination of the packet. + * + * A zone number of 0 is the global zone; a zone number of 0xffffffff + * means that the packet arrived from another host on the network, not + * from another zone on the same machine. + * + * An IPv4 or IPv6 datagram follows the pseudo-header; dli_family indicates + * which of those it is. + */ +#define DLT_IPNET		226 + +/* + * CAN (Controller Area Network) frames, with a pseudo-header as supplied + * by Linux SocketCAN, and with multi-byte numerical fields in that header + * in big-endian byte order. + * + * See Documentation/networking/can.txt in the Linux source. + * + * Requested by Felix Obenhuber <felix@obenhuber.de>. + */ +#define DLT_CAN_SOCKETCAN	227 + +/* + * Raw IPv4/IPv6; different from DLT_RAW in that the DLT_ value specifies + * whether it's v4 or v6.  Requested by Darren Reed <Darren.Reed@Sun.COM>. + */ +#define DLT_IPV4		228 +#define DLT_IPV6		229 + +/* + * IEEE 802.15.4, exactly as it appears in the spec (no padding, no + * nothing), and with no FCS at the end of the frame; requested by + * Jon Smirl <jonsmirl@gmail.com>. + */ +#define DLT_IEEE802_15_4_NOFCS	230 + +/* + * Raw D-Bus: + * + *	http://www.freedesktop.org/wiki/Software/dbus + * + * messages: + * + *	http://dbus.freedesktop.org/doc/dbus-specification.html#message-protocol-messages + * + * starting with the endianness flag, followed by the message type, etc., + * but without the authentication handshake before the message sequence: + * + *	http://dbus.freedesktop.org/doc/dbus-specification.html#auth-protocol + * + * Requested by Martin Vidner <martin@vidner.net>. + */ +#define DLT_DBUS		231 + +/* + * Juniper-private data link type, as per request from + * Hannes Gredler <hannes@juniper.net>. + */ +#define DLT_JUNIPER_VS			232 +#define DLT_JUNIPER_SRX_E2E		233 +#define DLT_JUNIPER_FIBRECHANNEL	234 + +/* + * DVB-CI (DVB Common Interface for communication between a PC Card + * module and a DVB receiver).  See + * + *	http://www.kaiser.cx/pcap-dvbci.html + * + * for the specification. + * + * Requested by Martin Kaiser <martin@kaiser.cx>. + */ +#define DLT_DVB_CI		235 + +/* + * Variant of 3GPP TS 27.010 multiplexing protocol (similar to, but + * *not* the same as, 27.010).  Requested by Hans-Christoph Schemmel + * <hans-christoph.schemmel@cinterion.com>. + */ +#define DLT_MUX27010		236 + +/* + * STANAG 5066 D_PDUs.  Requested by M. Baris Demiray + * <barisdemiray@gmail.com>. + */ +#define DLT_STANAG_5066_D_PDU	237 + +/* + * Juniper-private data link type, as per request from + * Hannes Gredler <hannes@juniper.net>. + */ +#define DLT_JUNIPER_ATM_CEMIC	238 + +/* + * NetFilter LOG messages + * (payload of netlink NFNL_SUBSYS_ULOG/NFULNL_MSG_PACKET packets) + * + * Requested by Jakub Zawadzki <darkjames-ws@darkjames.pl> + */ +#define DLT_NFLOG		239 + +/* + * Hilscher Gesellschaft fuer Systemautomation mbH link-layer type + * for Ethernet packets with a 4-byte pseudo-header and always + * with the payload including the FCS, as supplied by their + * netANALYZER hardware and software. + * + * Requested by Holger P. Frommer <HPfrommer@hilscher.com> + */ +#define DLT_NETANALYZER		240 + +/* + * Hilscher Gesellschaft fuer Systemautomation mbH link-layer type + * for Ethernet packets with a 4-byte pseudo-header and FCS and + * with the Ethernet header preceded by 7 bytes of preamble and + * 1 byte of SFD, as supplied by their netANALYZER hardware and + * software. + * + * Requested by Holger P. Frommer <HPfrommer@hilscher.com> + */ +#define DLT_NETANALYZER_TRANSPARENT	241 + +/* + * IP-over-InfiniBand, as specified by RFC 4391. + * + * Requested by Petr Sumbera <petr.sumbera@oracle.com>. + */ +#define DLT_IPOIB		242 + +/* + * MPEG-2 transport stream (ISO 13818-1/ITU-T H.222.0). + * + * Requested by Guy Martin <gmsoft@tuxicoman.be>. + */ +#define DLT_MPEG_2_TS		243 + +/* + * ng4T GmbH's UMTS Iub/Iur-over-ATM and Iub/Iur-over-IP format as + * used by their ng40 protocol tester. + * + * Requested by Jens Grimmer <jens.grimmer@ng4t.com>. + */ +#define DLT_NG40		244 + +/* + * Pseudo-header giving adapter number and flags, followed by an NFC + * (Near-Field Communications) Logical Link Control Protocol (LLCP) PDU, + * as specified by NFC Forum Logical Link Control Protocol Technical + * Specification LLCP 1.1. + * + * Requested by Mike Wakerly <mikey@google.com>. + */ +#define DLT_NFC_LLCP		245 + +/* + * 246 is used as LINKTYPE_PFSYNC; do not use it for any other purpose. + * + * DLT_PFSYNC has different values on different platforms, and all of + * them collide with something used elsewhere.  On platforms that + * don't already define it, define it as 246. + */ +#if !defined(__FreeBSD__) && !defined(__OpenBSD__) && !defined(__NetBSD__) && !defined(__DragonFly__) && !defined(__APPLE__) +#define DLT_PFSYNC		246 +#endif + +/* + * Raw InfiniBand packets, starting with the Local Routing Header. + * + * Requested by Oren Kladnitsky <orenk@mellanox.com>. + */ +#define DLT_INFINIBAND		247 + +/* + * SCTP, with no lower-level protocols (i.e., no IPv4 or IPv6). + * + * Requested by Michael Tuexen <Michael.Tuexen@lurchi.franken.de>. + */ +#define DLT_SCTP		248 + +/* + * USB packets, beginning with a USBPcap header. + * + * Requested by Tomasz Mon <desowin@gmail.com> + */ +#define DLT_USBPCAP		249 + +/* + * Schweitzer Engineering Laboratories "RTAC" product serial-line + * packets. + * + * Requested by Chris Bontje <chris_bontje@selinc.com>. + */ +#define DLT_RTAC_SERIAL		250 + +/* + * Bluetooth Low Energy air interface link-layer packets. + * + * Requested by Mike Kershaw <dragorn@kismetwireless.net>. + */ +#define DLT_BLUETOOTH_LE_LL	251 + +/* + * DLT type for upper-protocol layer PDU saves from wireshark. + * + * the actual contents are determined by two TAGs stored with each + * packet: + *   EXP_PDU_TAG_LINKTYPE          the link type (LINKTYPE_ value) of the + *				   original packet. + * + *   EXP_PDU_TAG_PROTO_NAME        the name of the wireshark dissector + * 				   that can make sense of the data stored. + */ +#define DLT_WIRESHARK_UPPER_PDU	252 + +/* + * DLT type for the netlink protocol (nlmon devices). + */ +#define DLT_NETLINK		253 + +/* + * Bluetooth Linux Monitor headers for the BlueZ stack. + */ +#define DLT_BLUETOOTH_LINUX_MONITOR	254 + +/* + * Bluetooth Basic Rate/Enhanced Data Rate baseband packets, as + * captured by Ubertooth. + */ +#define DLT_BLUETOOTH_BREDR_BB	255 + +/* + * Bluetooth Low Energy link layer packets, as captured by Ubertooth. + */ +#define DLT_BLUETOOTH_LE_LL_WITH_PHDR	256 + +/* + * PROFIBUS data link layer. + */ +#define DLT_PROFIBUS_DL		257 + +/* + * Apple's DLT_PKTAP headers. + * + * Sadly, the folks at Apple either had no clue that the DLT_USERn values + * are for internal use within an organization and partners only, and + * didn't know that the right way to get a link-layer header type is to + * ask tcpdump.org for one, or knew and didn't care, so they just + * used DLT_USER2, which causes problems for everything except for + * their version of tcpdump. + * + * So I'll just give them one; hopefully this will show up in a + * libpcap release in time for them to get this into 10.10 Big Sur + * or whatever Mavericks' successor is called.  LINKTYPE_PKTAP + * will be 258 *even on OS X*; that is *intentional*, so that + * PKTAP files look the same on *all* OSes (different OSes can have + * different numerical values for a given DLT_, but *MUST NOT* have + * different values for what goes in a file, as files can be moved + * between OSes!). + * + * When capturing, on a system with a Darwin-based OS, on a device + * that returns 149 (DLT_USER2 and Apple's DLT_PKTAP) with this + * version of libpcap, the DLT_ value for the pcap_t  will be DLT_PKTAP, + * and that will continue to be DLT_USER2 on Darwin-based OSes. That way, + * binary compatibility with Mavericks is preserved for programs using + * this version of libpcap.  This does mean that if you were using + * DLT_USER2 for some capture device on OS X, you can't do so with + * this version of libpcap, just as you can't with Apple's libpcap - + * on OS X, they define DLT_PKTAP to be DLT_USER2, so programs won't + * be able to distinguish between PKTAP and whatever you were using + * DLT_USER2 for. + * + * If the program saves the capture to a file using this version of + * libpcap's pcap_dump code, the LINKTYPE_ value in the file will be + * LINKTYPE_PKTAP, which will be 258, even on Darwin-based OSes. + * That way, the file will *not* be a DLT_USER2 file.  That means + * that the latest version of tcpdump, when built with this version + * of libpcap, and sufficiently recent versions of Wireshark will + * be able to read those files and interpret them correctly; however, + * Apple's version of tcpdump in OS X 10.9 won't be able to handle + * them.  (Hopefully, Apple will pick up this version of libpcap, + * and the corresponding version of tcpdump, so that tcpdump will + * be able to handle the old LINKTYPE_USER2 captures *and* the new + * LINKTYPE_PKTAP captures.) + */ +#ifdef __APPLE__ +#define DLT_PKTAP	DLT_USER2 +#else +#define DLT_PKTAP	258 +#endif + +/* + * Ethernet packets preceded by a header giving the last 6 octets + * of the preamble specified by 802.3-2012 Clause 65, section + * 65.1.3.2 "Transmit". + */ +#define DLT_EPON	259 + +/* + * IPMI trace packets, as specified by Table 3-20 "Trace Data Block Format" + * in the PICMG HPM.2 specification. + */ +#define DLT_IPMI_HPM_2	260 + +/* + * per  Joshua Wright <jwright@hasborg.com>, formats for Zwave captures. + */ +#define DLT_ZWAVE_R1_R2  261 +#define DLT_ZWAVE_R3     262 + +/* + * per Steve Karg <skarg@users.sourceforge.net>, formats for Wattstopper + * Digital Lighting Management room bus serial protocol captures. + */ +#define DLT_WATTSTOPPER_DLM     263 + +/* + * ISO 14443 contactless smart card messages. + */ +#define DLT_ISO_14443	264 + +/* + * Radio data system (RDS) groups.  IEC 62106. + * Per Jonathan Brucker <jonathan.brucke@gmail.com>. + */ +#define DLT_RDS		265 + +/* + * USB packets, beginning with a Darwin (macOS, etc.) header. + */ +#define DLT_USB_DARWIN	266 + +/* + * OpenBSD DLT_OPENFLOW. + */ +#define DLT_OPENFLOW	267 + +/* + * SDLC frames containing SNA PDUs. + */ +#define DLT_SDLC	268 + +/* + * per "Selvig, Bjorn" <b.selvig@ti.com> used for + * TI protocol sniffer. + */ +#define DLT_TI_LLN_SNIFFER	269 + +/* + * per: Erik de Jong <erikdejong at gmail.com> for + *   https://github.com/eriknl/LoRaTap/releases/tag/v0.1 + */ +#define DLT_LORATAP             270 + +/* + * per: Stefanha at gmail.com for + *   http://lists.sandelman.ca/pipermail/tcpdump-workers/2017-May/000772.html + * and: https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/tree/include/uapi/linux/vsockmon.h + * for: http://qemu-project.org/Features/VirtioVsock + */ +#define DLT_VSOCK               271 + +/* + * Nordic Semiconductor Bluetooth LE sniffer. + */ +#define DLT_NORDIC_BLE		272 + +/* + * Excentis DOCSIS 3.1 RF sniffer (XRA-31) + *   per: bruno.verstuyft at excentis.com + *        http://www.xra31.com/xra-header + */ +#define DLT_DOCSIS31_XRA31	273 + +/* + * mPackets, as specified by IEEE 802.3br Figure 99-4, starting + * with the preamble and always ending with a CRC field. + */ +#define DLT_ETHERNET_MPACKET	274 + +/* + * In case the code that includes this file (directly or indirectly) + * has also included OS files that happen to define DLT_MATCHING_MAX, + * with a different value (perhaps because that OS hasn't picked up + * the latest version of our DLT definitions), we undefine the + * previous value of DLT_MATCHING_MAX. + */ +#ifdef DLT_MATCHING_MAX +#undef DLT_MATCHING_MAX +#endif +#define DLT_MATCHING_MAX	274	/* highest value in the "matching" range */ + +/* + * DLT and savefile link type values are split into a class and + * a member of that class.  A class value of 0 indicates a regular + * DLT_/LINKTYPE_ value. + */ +#define DLT_CLASS(x)		((x) & 0x03ff0000) + +/* + * NetBSD-specific generic "raw" link type.  The class value indicates + * that this is the generic raw type, and the lower 16 bits are the + * address family we're dealing with.  Those values are NetBSD-specific; + * do not assume that they correspond to AF_ values for your operating + * system. + */ +#define	DLT_CLASS_NETBSD_RAWAF	0x02240000 +#define	DLT_NETBSD_RAWAF(af)	(DLT_CLASS_NETBSD_RAWAF | (af)) +#define	DLT_NETBSD_RAWAF_AF(x)	((x) & 0x0000ffff) +#define	DLT_IS_NETBSD_RAWAF(x)	(DLT_CLASS(x) == DLT_CLASS_NETBSD_RAWAF) + +#endif /* !defined(lib_pcap_dlt_h) */ diff --git a/src/frontend/qt_sdl/pcap/funcattrs.h b/src/frontend/qt_sdl/pcap/funcattrs.h new file mode 100644 index 0000000..d1cc2bf --- /dev/null +++ b/src/frontend/qt_sdl/pcap/funcattrs.h @@ -0,0 +1,242 @@ +/* -*- Mode: c; tab-width: 8; indent-tabs-mode: 1; c-basic-offset: 8; -*- */ +/* + * Copyright (c) 1993, 1994, 1995, 1996, 1997 + *	The Regents of the University of California.  All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + *    must display the following acknowledgement: + *	This product includes software developed by the Computer Systems + *	Engineering Group at Lawrence Berkeley Laboratory. + * 4. Neither the name of the University nor of the Laboratory may be used + *    to endorse or promote products derived from this software without + *    specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +#ifndef lib_pcap_funcattrs_h +#define lib_pcap_funcattrs_h + +#include <pcap/compiler-tests.h> + +/* + * Attributes to apply to functions and their arguments, using various + * compiler-specific extensions. + */ + +/* + * PCAP_API_DEF must be used when defining *data* exported from + * libpcap.  It can be used when defining *functions* exported + * from libpcap, but it doesn't have to be used there.  It + * should not be used in declarations in headers. + * + * PCAP_API must be used when *declaring* data or functions + * exported from libpcap; PCAP_API_DEF won't work on all platforms. + */ + +#if defined(_WIN32) +  /* +   * For Windows: +   * +   *    when building libpcap: +   * +   *       if we're building it as a DLL, we have to declare API +   *       functions with __declspec(dllexport); +   * +   *       if we're building it as a static library, we don't want +   *       to do so. +   * +   *    when using libpcap: +   * +   *       if we're using the DLL, calls to its functions are a +   *       little more efficient if they're declared with +   *       __declspec(dllimport); +   * +   *       if we're not using the dll, we don't want to declare +   *       them that way. +   * +   * So: +   * +   *    if pcap_EXPORTS is defined, we define PCAP_API_DEF as +   *     __declspec(dllexport); +   * +   *    if PCAP_DLL is defined, we define PCAP_API_DEF as +   *    __declspec(dllimport); +   * +   *    otherwise, we define PCAP_API_DEF as nothing. +   */ +  #if defined(pcap_EXPORTS) +    /* +     * We're compiling libpcap, so we should export functions in our +     * API. +     */ +    #define PCAP_API_DEF	__declspec(dllexport) +  #elif defined(PCAP_DLL) +    #define PCAP_API_DEF	__declspec(dllimport) +  #else +    #define PCAP_API_DEF +  #endif +#elif defined(MSDOS) +  /* XXX - does this need special treatment? */ +  #define PCAP_API_DEF +#else /* UN*X */ +  #ifdef pcap_EXPORTS +    /* +     * We're compiling libpcap, so we should export functions in our API. +     * The compiler might be configured not to export functions from a +     * shared library by default, so we might have to explicitly mark +     * functions as exported. +     */ +    #if PCAP_IS_AT_LEAST_GNUC_VERSION(3,4) \ +        || PCAP_IS_AT_LEAST_XL_C_VERSION(12,0) +      /* +       * GCC 3.4 or later, or some compiler asserting compatibility with +       * GCC 3.4 or later, or XL C 13.0 or later, so we have +       * __attribute__((visibility()). +       */ +      #define PCAP_API_DEF	__attribute__((visibility("default"))) +    #elif PCAP_IS_AT_LEAST_SUNC_VERSION(5,5) +      /* +       * Sun C 5.5 or later, so we have __global. +       * (Sun C 5.9 and later also have __attribute__((visibility()), +       * but there's no reason to prefer it with Sun C.) +       */ +      #define PCAP_API_DEF	__global +    #else +      /* +       * We don't have anything to say. +       */ +      #define PCAP_API_DEF +    #endif +  #else +    /* +     * We're not building libpcap. +     */ +    #define PCAP_API_DEF +  #endif +#endif /* _WIN32/MSDOS/UN*X */ + +#define PCAP_API	PCAP_API_DEF extern + +/* + * PCAP_NORETURN, before a function declaration, means "this function + * never returns".  (It must go before the function declaration, e.g. + * "extern PCAP_NORETURN func(...)" rather than after the function + * declaration, as the MSVC version has to go before the declaration.) + */ +#if __has_attribute(noreturn) \ +    || PCAP_IS_AT_LEAST_GNUC_VERSION(2,5) \ +    || PCAP_IS_AT_LEAST_SUNC_VERSION(5,9) \ +    || PCAP_IS_AT_LEAST_XL_C_VERSION(10,1) \ +    || PCAP_IS_AT_LEAST_HP_C_VERSION(6,10) +  /* +   * Compiler with support for __attribute((noreturn)), or GCC 2.5 and +   * later, or Solaris Studio 12 (Sun C 5.9) and later, or IBM XL C 10.1 +   * and later (do any earlier versions of XL C support this?), or +   * HP aCC A.06.10 and later. +   */ +  #define PCAP_NORETURN __attribute((noreturn)) +#elif defined(_MSC_VER) +  /* +   * MSVC. +   */ +  #define PCAP_NORETURN __declspec(noreturn) +#else +  #define PCAP_NORETURN +#endif + +/* + * PCAP_PRINTFLIKE(x,y), after a function declaration, means "this function + * does printf-style formatting, with the xth argument being the format + * string and the yth argument being the first argument for the format + * string". + */ +#if __has_attribute(__format__) \ +    || PCAP_IS_AT_LEAST_GNUC_VERSION(2,3) \ +    || PCAP_IS_AT_LEAST_XL_C_VERSION(10,1) \ +    || PCAP_IS_AT_LEAST_HP_C_VERSION(6,10) +  /* +   * Compiler with support for it, or GCC 2.3 and later, or IBM XL C 10.1 +   * and later (do any earlier versions of XL C support this?), +   * or HP aCC A.06.10 and later. +   */ +  #define PCAP_PRINTFLIKE(x,y) __attribute__((__format__(__printf__,x,y))) +#else +  #define PCAP_PRINTFLIKE(x,y) +#endif + +/* + * PCAP_DEPRECATED(func, msg), after a function declaration, marks the + * function as deprecated. + * + * The first argument is the name of the function; the second argument is + * a string giving the warning message to use if the compiler supports that. + * + * (Thank you, Microsoft, for requiring the function name.) + */ +#if __has_attribute(deprecated) \ +    || PCAP_IS_AT_LEAST_GNUC_VERSION(4,5) \ +    || PCAP_IS_AT_LEAST_SUNC_VERSION(5,13) +  /* +   * Compiler that supports __has_attribute and __attribute__((deprecated)), +   * or GCC 4.5 and later, or Sun/Oracle C 12.4 (Sun C 5.13) or later. +   * +   * Those support __attribute__((deprecated(msg))) (we assume, perhaps +   * incorrectly, that anything that supports __has_attribute() is +   * recent enough to support __attribute__((deprecated(msg)))). +   */ +  #define PCAP_DEPRECATED(func, msg)	__attribute__((deprecated(msg))) +#elif PCAP_IS_AT_LEAST_GNUC_VERSION(3,1) +  /* +   * GCC 3.1 through 4.4. +   * +   * Those support __attribute__((deprecated)) but not +   * __attribute__((deprecated(msg))). +   */ +  #define PCAP_DEPRECATED(func, msg)	__attribute__((deprecated)) +#elif (defined(_MSC_VER) && (_MSC_VER >= 1500)) && !defined(pcap_EXPORTS) +  /* +   * MSVC from Visual Studio 2008 or later, and we're not building +   * libpcap itself. +   * +   * If we *are* building libpcap, we don't want this, as it'll warn +   * us even if we *define* the function. +   */ +  #define PCAP_DEPRECATED(func, msg)	__pragma(deprecated(func)) +#else +  #define PCAP_DEPRECATED(func, msg) +#endif + +/* + * For flagging arguments as format strings in MSVC. + */ +#ifdef _MSC_VER + #include <sal.h> + #if _MSC_VER > 1400 +  #define PCAP_FORMAT_STRING(p) _Printf_format_string_ p + #else +  #define PCAP_FORMAT_STRING(p) __format_string p + #endif +#else + #define PCAP_FORMAT_STRING(p) p +#endif + +#endif /* lib_pcap_funcattrs_h */ diff --git a/src/frontend/qt_sdl/pcap/ipnet.h b/src/frontend/qt_sdl/pcap/ipnet.h new file mode 100644 index 0000000..5330847 --- /dev/null +++ b/src/frontend/qt_sdl/pcap/ipnet.h @@ -0,0 +1,43 @@ +/*- + * Copyright (c) 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997 + *	The Regents of the University of California.  All rights reserved. + * + * This code is derived from the Stanford/CMU enet packet filter, + * (net/enet.c) distributed as part of 4.3BSD, and code contributed + * to Berkeley by Steven McCanne and Van Jacobson both of Lawrence + * Berkeley Laboratory. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + *    must display the following acknowledgement: + *      This product includes software developed by the University of + *      California, Berkeley and its contributors. + * 4. Neither the name of the University nor the names of its contributors + *    may be used to endorse or promote products derived from this software + *    without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +#define	IPH_AF_INET	2		/* Matches Solaris's AF_INET */ +#define	IPH_AF_INET6	26		/* Matches Solaris's AF_INET6 */ + +#define	IPNET_OUTBOUND		1 +#define	IPNET_INBOUND		2 diff --git a/src/frontend/qt_sdl/pcap/namedb.h b/src/frontend/qt_sdl/pcap/namedb.h new file mode 100644 index 0000000..73fb40a --- /dev/null +++ b/src/frontend/qt_sdl/pcap/namedb.h @@ -0,0 +1,85 @@ +/* + * Copyright (c) 1994, 1996 + *	The Regents of the University of California.  All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + *    must display the following acknowledgement: + *	This product includes software developed by the Computer Systems + *	Engineering Group at Lawrence Berkeley Laboratory. + * 4. Neither the name of the University nor of the Laboratory may be used + *    to endorse or promote products derived from this software without + *    specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +#ifndef lib_pcap_namedb_h +#define lib_pcap_namedb_h + +#ifdef __cplusplus +extern "C" { +#endif + +/* + * As returned by the pcap_next_etherent() + * XXX this stuff doesn't belong in this interface, but this + * library already must do name to address translation, so + * on systems that don't have support for /etc/ethers, we + * export these hooks since they're already being used by + * some applications (such as tcpdump) and already being + * marked as exported in some OSes offering libpcap (such + * as Debian). + */ +struct pcap_etherent { +	u_char addr[6]; +	char name[122]; +}; +#ifndef PCAP_ETHERS_FILE +#define PCAP_ETHERS_FILE "/etc/ethers" +#endif +PCAP_API struct	pcap_etherent *pcap_next_etherent(FILE *); +PCAP_API u_char *pcap_ether_hostton(const char*); +PCAP_API u_char *pcap_ether_aton(const char *); + +PCAP_API bpf_u_int32 **pcap_nametoaddr(const char *); +#ifdef INET6 +PCAP_API struct addrinfo *pcap_nametoaddrinfo(const char *); +#endif +PCAP_API bpf_u_int32 pcap_nametonetaddr(const char *); + +PCAP_API int	pcap_nametoport(const char *, int *, int *); +PCAP_API int	pcap_nametoportrange(const char *, int *, int *, int *); +PCAP_API int	pcap_nametoproto(const char *); +PCAP_API int	pcap_nametoeproto(const char *); +PCAP_API int	pcap_nametollc(const char *); +/* + * If a protocol is unknown, PROTO_UNDEF is returned. + * Also, pcap_nametoport() returns the protocol along with the port number. + * If there are ambiguous entried in /etc/services (i.e. domain + * can be either tcp or udp) PROTO_UNDEF is returned. + */ +#define PROTO_UNDEF		-1 + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/src/frontend/qt_sdl/pcap/nflog.h b/src/frontend/qt_sdl/pcap/nflog.h new file mode 100644 index 0000000..29a71d2 --- /dev/null +++ b/src/frontend/qt_sdl/pcap/nflog.h @@ -0,0 +1,94 @@ +/* + * Copyright (c) 2013, Petar Alilovic, + * Faculty of Electrical Engineering and Computing, University of Zagreb + * All rights reserved + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright notice, + *	 this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + *	 notice, this list of conditions and the following disclaimer in the + *	 documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY + * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR ANY + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER + * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH + * DAMAGE. + */ + +#ifndef lib_pcap_nflog_h +#define lib_pcap_nflog_h + +#include <pcap/pcap-inttypes.h> + +/* + * Structure of an NFLOG header and TLV parts, as described at + * http://www.tcpdump.org/linktypes/LINKTYPE_NFLOG.html + * + * The NFLOG header is big-endian. + * + * The TLV length and type are in host byte order.  The value is either + * big-endian or is an array of bytes in some externally-specified byte + * order (text string, link-layer address, link-layer header, packet + * data, etc.). + */ +typedef struct nflog_hdr { +	uint8_t	nflog_family;		/* address family */ +	uint8_t	nflog_version;		/* version */ +	uint16_t	nflog_rid;		/* resource ID */ +} nflog_hdr_t; + +typedef struct nflog_tlv { +	uint16_t	tlv_length;		/* tlv length */ +	uint16_t	tlv_type;		/* tlv type */ +	/* value follows this */ +} nflog_tlv_t; + +typedef struct nflog_packet_hdr { +	uint16_t	hw_protocol;	/* hw protocol */ +	uint8_t	hook;		/* netfilter hook */ +	uint8_t	pad;		/* padding to 32 bits */ +} nflog_packet_hdr_t; + +typedef struct nflog_hwaddr { +	uint16_t	hw_addrlen;	/* address length */ +	uint16_t	pad;		/* padding to 32-bit boundary */ +	uint8_t	hw_addr[8];	/* address, up to 8 bytes */ +} nflog_hwaddr_t; + +typedef struct nflog_timestamp { +	uint64_t	sec; +	uint64_t	usec; +} nflog_timestamp_t; + +/* + * TLV types. + */ +#define NFULA_PACKET_HDR		1	/* nflog_packet_hdr_t */ +#define NFULA_MARK			2	/* packet mark from skbuff */ +#define NFULA_TIMESTAMP			3	/* nflog_timestamp_t for skbuff's time stamp */ +#define NFULA_IFINDEX_INDEV		4	/* ifindex of device on which packet received (possibly bridge group) */ +#define NFULA_IFINDEX_OUTDEV		5	/* ifindex of device on which packet transmitted (possibly bridge group) */ +#define NFULA_IFINDEX_PHYSINDEV		6	/* ifindex of physical device on which packet received (not bridge group) */ +#define NFULA_IFINDEX_PHYSOUTDEV	7	/* ifindex of physical device on which packet transmitted (not bridge group) */ +#define NFULA_HWADDR			8	/* nflog_hwaddr_t for hardware address */ +#define NFULA_PAYLOAD			9	/* packet payload */ +#define NFULA_PREFIX			10	/* text string - null-terminated, count includes NUL */ +#define NFULA_UID			11	/* UID owning socket on which packet was sent/received */ +#define NFULA_SEQ			12	/* sequence number of packets on this NFLOG socket */ +#define NFULA_SEQ_GLOBAL		13	/* sequence number of pakets on all NFLOG sockets */ +#define NFULA_GID			14	/* GID owning socket on which packet was sent/received */ +#define NFULA_HWTYPE			15	/* ARPHRD_ type of skbuff's device */ +#define NFULA_HWHEADER			16	/* skbuff's MAC-layer header */ +#define NFULA_HWLEN			17	/* length of skbuff's MAC-layer header */ + +#endif diff --git a/src/frontend/qt_sdl/pcap/pcap-inttypes.h b/src/frontend/qt_sdl/pcap/pcap-inttypes.h new file mode 100644 index 0000000..af2c23c --- /dev/null +++ b/src/frontend/qt_sdl/pcap/pcap-inttypes.h @@ -0,0 +1,117 @@ +/* + * Copyright (c) 2002 - 2005 NetGroup, Politecnico di Torino (Italy) + * Copyright (c) 2005 - 2009 CACE Technologies, Inc. Davis (California) + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. Neither the name of the Politecnico di Torino nor the names of its + * contributors may be used to endorse or promote products derived from + * this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ +#ifndef pcap_pcap_inttypes_h +#define pcap_pcap_inttypes_h + +/* + * Get the integer types and PRi[doux]64 values from C99 <inttypes.h> + * defined, by hook or by crook. + */ +#if defined(_MSC_VER) +  /* +   * Compiler is MSVC. +   */ +  #if _MSC_VER >= 1800 +    /* +     * VS 2013 or newer; we have <inttypes.h>. +     */ +    #include <inttypes.h> +  #else +    /* +     * Earlier VS; we have to define this stuff ourselves. +     */ +    typedef unsigned char uint8_t; +    typedef signed char int8_t; +    typedef unsigned short uint16_t; +    typedef signed short int16_t; +    typedef unsigned int uint32_t; +    typedef signed int int32_t; +    #ifdef _MSC_EXTENSIONS +      typedef unsigned _int64 uint64_t; +      typedef _int64 int64_t; +    #else /* _MSC_EXTENSIONS */ +      typedef unsigned long long uint64_t; +      typedef long long int64_t; +    #endif +  #endif + +  /* +   * These may be defined by <inttypes.h>. +   * +   * XXX - for MSVC, we always want the _MSC_EXTENSIONS versions. +   * What about other compilers?  If, as the MinGW Web site says MinGW +   * does, the other compilers just use Microsoft's run-time library, +   * then they should probably use the _MSC_EXTENSIONS even if the +   * compiler doesn't define _MSC_EXTENSIONS. +   * +   * XXX - we currently aren't using any of these, but this allows +   * their use in the future. +   */ +  #ifndef PRId64 +    #ifdef _MSC_EXTENSIONS +      #define PRId64	"I64d" +    #else +      #define PRId64	"lld" +    #endif +  #endif /* PRId64 */ + +  #ifndef PRIo64 +    #ifdef _MSC_EXTENSIONS +      #define PRIo64	"I64o" +    #else +      #define PRIo64	"llo" +    #endif +  #endif /* PRIo64 */ + +  #ifndef PRIx64 +    #ifdef _MSC_EXTENSIONS +      #define PRIx64	"I64x" +    #else +      #define PRIx64	"llx" +    #endif +  #endif + +  #ifndef PRIu64 +    #ifdef _MSC_EXTENSIONS +      #define PRIu64	"I64u" +    #else +      #define PRIu64	"llu" +    #endif +  #endif +#elif defined(__MINGW32__) || !defined(_WIN32) +  /* +   * Compiler is MinGW or target is UN*X or MS-DOS.  Just use +   * <inttypes.h>. +   */ +  #include <inttypes.h> +#endif + +#endif /* pcap/pcap-inttypes.h */ diff --git a/src/frontend/qt_sdl/pcap/pcap.h b/src/frontend/qt_sdl/pcap/pcap.h new file mode 100644 index 0000000..d32e2a9 --- /dev/null +++ b/src/frontend/qt_sdl/pcap/pcap.h @@ -0,0 +1,966 @@ +/* -*- Mode: c; tab-width: 8; indent-tabs-mode: 1; c-basic-offset: 8; -*- */ +/* + * Copyright (c) 1993, 1994, 1995, 1996, 1997 + *	The Regents of the University of California.  All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + *    must display the following acknowledgement: + *	This product includes software developed by the Computer Systems + *	Engineering Group at Lawrence Berkeley Laboratory. + * 4. Neither the name of the University nor of the Laboratory may be used + *    to endorse or promote products derived from this software without + *    specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +/* + * Remote packet capture mechanisms and extensions from WinPcap: + * + * Copyright (c) 2002 - 2003 + * NetGroup, Politecnico di Torino (Italy) + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. Neither the name of the Politecnico di Torino nor the names of its + * contributors may be used to endorse or promote products derived from + * this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef lib_pcap_pcap_h +#define lib_pcap_pcap_h + +#include <pcap/funcattrs.h> + +#include <pcap/pcap-inttypes.h> + +#if defined(_WIN32) +  #include <winsock2.h>		/* u_int, u_char etc. */ +  #include <io.h>		/* _get_osfhandle() */ +#elif defined(MSDOS) +  #include <sys/types.h>	/* u_int, u_char etc. */ +  #include <sys/socket.h> +#else /* UN*X */ +  #include <sys/types.h>	/* u_int, u_char etc. */ +  #include <sys/time.h> +#endif /* _WIN32/MSDOS/UN*X */ + +#ifndef PCAP_DONT_INCLUDE_PCAP_BPF_H +#include <pcap/bpf.h> +#endif + +#include <stdio.h> + +#ifdef __cplusplus +extern "C" { +#endif + +/* + * Version number of the current version of the pcap file format. + * + * NOTE: this is *NOT* the version number of the libpcap library. + * To fetch the version information for the version of libpcap + * you're using, use pcap_lib_version(). + */ +#define PCAP_VERSION_MAJOR 2 +#define PCAP_VERSION_MINOR 4 + +#define PCAP_ERRBUF_SIZE 256 + +/* + * Compatibility for systems that have a bpf.h that + * predates the bpf typedefs for 64-bit support. + */ +#if BPF_RELEASE - 0 < 199406 +typedef	int bpf_int32; +typedef	u_int bpf_u_int32; +#endif + +typedef struct pcap pcap_t; +typedef struct pcap_dumper pcap_dumper_t; +typedef struct pcap_if pcap_if_t; +typedef struct pcap_addr pcap_addr_t; + +/* + * The first record in the file contains saved values for some + * of the flags used in the printout phases of tcpdump. + * Many fields here are 32 bit ints so compilers won't insert unwanted + * padding; these files need to be interchangeable across architectures. + * + * Do not change the layout of this structure, in any way (this includes + * changes that only affect the length of fields in this structure). + * + * Also, do not change the interpretation of any of the members of this + * structure, in any way (this includes using values other than + * LINKTYPE_ values, as defined in "savefile.c", in the "linktype" + * field). + * + * Instead: + * + *	introduce a new structure for the new format, if the layout + *	of the structure changed; + * + *	send mail to "tcpdump-workers@lists.tcpdump.org", requesting + *	a new magic number for your new capture file format, and, when + *	you get the new magic number, put it in "savefile.c"; + * + *	use that magic number for save files with the changed file + *	header; + * + *	make the code in "savefile.c" capable of reading files with + *	the old file header as well as files with the new file header + *	(using the magic number to determine the header format). + * + * Then supply the changes by forking the branch at + * + *	https://github.com/the-tcpdump-group/libpcap/issues + * + * and issuing a pull request, so that future versions of libpcap and + * programs that use it (such as tcpdump) will be able to read your new + * capture file format. + */ +struct pcap_file_header { +	bpf_u_int32 magic; +	u_short version_major; +	u_short version_minor; +	bpf_int32 thiszone;	/* gmt to local correction */ +	bpf_u_int32 sigfigs;	/* accuracy of timestamps */ +	bpf_u_int32 snaplen;	/* max length saved portion of each pkt */ +	bpf_u_int32 linktype;	/* data link type (LINKTYPE_*) */ +}; + +/* + * Macros for the value returned by pcap_datalink_ext(). + * + * If LT_FCS_LENGTH_PRESENT(x) is true, the LT_FCS_LENGTH(x) macro + * gives the FCS length of packets in the capture. + */ +#define LT_FCS_LENGTH_PRESENT(x)	((x) & 0x04000000) +#define LT_FCS_LENGTH(x)		(((x) & 0xF0000000) >> 28) +#define LT_FCS_DATALINK_EXT(x)		((((x) & 0xF) << 28) | 0x04000000) + +typedef enum { +       PCAP_D_INOUT = 0, +       PCAP_D_IN, +       PCAP_D_OUT +} pcap_direction_t; + +/* + * Generic per-packet information, as supplied by libpcap. + * + * The time stamp can and should be a "struct timeval", regardless of + * whether your system supports 32-bit tv_sec in "struct timeval", + * 64-bit tv_sec in "struct timeval", or both if it supports both 32-bit + * and 64-bit applications.  The on-disk format of savefiles uses 32-bit + * tv_sec (and tv_usec); this structure is irrelevant to that.  32-bit + * and 64-bit versions of libpcap, even if they're on the same platform, + * should supply the appropriate version of "struct timeval", even if + * that's not what the underlying packet capture mechanism supplies. + */ +struct pcap_pkthdr { +	struct timeval ts;	/* time stamp */ +	bpf_u_int32 caplen;	/* length of portion present */ +	bpf_u_int32 len;	/* length this packet (off wire) */ +}; + +/* + * As returned by the pcap_stats() + */ +struct pcap_stat { +	u_int ps_recv;		/* number of packets received */ +	u_int ps_drop;		/* number of packets dropped */ +	u_int ps_ifdrop;	/* drops by interface -- only supported on some platforms */ +#ifdef _WIN32 +	u_int ps_capt;		/* number of packets that reach the application */ +	u_int ps_sent;		/* number of packets sent by the server on the network */ +	u_int ps_netdrop;	/* number of packets lost on the network */ +#endif /* _WIN32 */ +}; + +#ifdef MSDOS +/* + * As returned by the pcap_stats_ex() + */ +struct pcap_stat_ex { +       u_long  rx_packets;        /* total packets received       */ +       u_long  tx_packets;        /* total packets transmitted    */ +       u_long  rx_bytes;          /* total bytes received         */ +       u_long  tx_bytes;          /* total bytes transmitted      */ +       u_long  rx_errors;         /* bad packets received         */ +       u_long  tx_errors;         /* packet transmit problems     */ +       u_long  rx_dropped;        /* no space in Rx buffers       */ +       u_long  tx_dropped;        /* no space available for Tx    */ +       u_long  multicast;         /* multicast packets received   */ +       u_long  collisions; + +       /* detailed rx_errors: */ +       u_long  rx_length_errors; +       u_long  rx_over_errors;    /* receiver ring buff overflow  */ +       u_long  rx_crc_errors;     /* recv'd pkt with crc error    */ +       u_long  rx_frame_errors;   /* recv'd frame alignment error */ +       u_long  rx_fifo_errors;    /* recv'r fifo overrun          */ +       u_long  rx_missed_errors;  /* recv'r missed packet         */ + +       /* detailed tx_errors */ +       u_long  tx_aborted_errors; +       u_long  tx_carrier_errors; +       u_long  tx_fifo_errors; +       u_long  tx_heartbeat_errors; +       u_long  tx_window_errors; +     }; +#endif + +/* + * Item in a list of interfaces. + */ +struct pcap_if { +	struct pcap_if *next; +	char *name;		/* name to hand to "pcap_open_live()" */ +	char *description;	/* textual description of interface, or NULL */ +	struct pcap_addr *addresses; +	bpf_u_int32 flags;	/* PCAP_IF_ interface flags */ +}; + +#define PCAP_IF_LOOPBACK	0x00000001	/* interface is loopback */ +#define PCAP_IF_UP		0x00000002	/* interface is up */ +#define PCAP_IF_RUNNING		0x00000004	/* interface is running */ + +/* + * Representation of an interface address. + */ +struct pcap_addr { +	struct pcap_addr *next; +	struct sockaddr *addr;		/* address */ +	struct sockaddr *netmask;	/* netmask for that address */ +	struct sockaddr *broadaddr;	/* broadcast address for that address */ +	struct sockaddr *dstaddr;	/* P2P destination address for that address */ +}; + +typedef void (*pcap_handler)(u_char *, const struct pcap_pkthdr *, +			     const u_char *); + +/* + * Error codes for the pcap API. + * These will all be negative, so you can check for the success or + * failure of a call that returns these codes by checking for a + * negative value. + */ +#define PCAP_ERROR			-1	/* generic error code */ +#define PCAP_ERROR_BREAK		-2	/* loop terminated by pcap_breakloop */ +#define PCAP_ERROR_NOT_ACTIVATED	-3	/* the capture needs to be activated */ +#define PCAP_ERROR_ACTIVATED		-4	/* the operation can't be performed on already activated captures */ +#define PCAP_ERROR_NO_SUCH_DEVICE	-5	/* no such device exists */ +#define PCAP_ERROR_RFMON_NOTSUP		-6	/* this device doesn't support rfmon (monitor) mode */ +#define PCAP_ERROR_NOT_RFMON		-7	/* operation supported only in monitor mode */ +#define PCAP_ERROR_PERM_DENIED		-8	/* no permission to open the device */ +#define PCAP_ERROR_IFACE_NOT_UP		-9	/* interface isn't up */ +#define PCAP_ERROR_CANTSET_TSTAMP_TYPE	-10	/* this device doesn't support setting the time stamp type */ +#define PCAP_ERROR_PROMISC_PERM_DENIED	-11	/* you don't have permission to capture in promiscuous mode */ +#define PCAP_ERROR_TSTAMP_PRECISION_NOTSUP -12  /* the requested time stamp precision is not supported */ + +/* + * Warning codes for the pcap API. + * These will all be positive and non-zero, so they won't look like + * errors. + */ +#define PCAP_WARNING			1	/* generic warning code */ +#define PCAP_WARNING_PROMISC_NOTSUP	2	/* this device doesn't support promiscuous mode */ +#define PCAP_WARNING_TSTAMP_TYPE_NOTSUP	3	/* the requested time stamp type is not supported */ + +/* + * Value to pass to pcap_compile() as the netmask if you don't know what + * the netmask is. + */ +#define PCAP_NETMASK_UNKNOWN	0xffffffff + +/* + * We're deprecating pcap_lookupdev() for various reasons (not + * thread-safe, can behave weirdly with WinPcap).  Callers + * should use pcap_findalldevs() and use the first device. + */ +PCAP_API char	*pcap_lookupdev(char *) +PCAP_DEPRECATED(pcap_lookupdev, "use 'pcap_findalldevs' and use the first device"); + +PCAP_API int	pcap_lookupnet(const char *, bpf_u_int32 *, bpf_u_int32 *, char *); + +PCAP_API pcap_t	*pcap_create(const char *, char *); +PCAP_API int	pcap_set_snaplen(pcap_t *, int); +PCAP_API int	pcap_set_promisc(pcap_t *, int); +PCAP_API int	pcap_can_set_rfmon(pcap_t *); +PCAP_API int	pcap_set_rfmon(pcap_t *, int); +PCAP_API int	pcap_set_timeout(pcap_t *, int); +PCAP_API int	pcap_set_tstamp_type(pcap_t *, int); +PCAP_API int	pcap_set_immediate_mode(pcap_t *, int); +PCAP_API int	pcap_set_buffer_size(pcap_t *, int); +PCAP_API int	pcap_set_tstamp_precision(pcap_t *, int); +PCAP_API int	pcap_get_tstamp_precision(pcap_t *); +PCAP_API int	pcap_activate(pcap_t *); + +PCAP_API int	pcap_list_tstamp_types(pcap_t *, int **); +PCAP_API void	pcap_free_tstamp_types(int *); +PCAP_API int	pcap_tstamp_type_name_to_val(const char *); +PCAP_API const char *pcap_tstamp_type_val_to_name(int); +PCAP_API const char *pcap_tstamp_type_val_to_description(int); + +#ifdef __linux__ +PCAP_API int	pcap_set_protocol(pcap_t *, int); +#endif + +/* + * Time stamp types. + * Not all systems and interfaces will necessarily support all of these. + * + * A system that supports PCAP_TSTAMP_HOST is offering time stamps + * provided by the host machine, rather than by the capture device, + * but not committing to any characteristics of the time stamp; + * it will not offer any of the PCAP_TSTAMP_HOST_ subtypes. + * + * PCAP_TSTAMP_HOST_LOWPREC is a time stamp, provided by the host machine, + * that's low-precision but relatively cheap to fetch; it's normally done + * using the system clock, so it's normally synchronized with times you'd + * fetch from system calls. + * + * PCAP_TSTAMP_HOST_HIPREC is a time stamp, provided by the host machine, + * that's high-precision; it might be more expensive to fetch.  It might + * or might not be synchronized with the system clock, and might have + * problems with time stamps for packets received on different CPUs, + * depending on the platform. + * + * PCAP_TSTAMP_ADAPTER is a high-precision time stamp supplied by the + * capture device; it's synchronized with the system clock. + * + * PCAP_TSTAMP_ADAPTER_UNSYNCED is a high-precision time stamp supplied by + * the capture device; it's not synchronized with the system clock. + * + * Note that time stamps synchronized with the system clock can go + * backwards, as the system clock can go backwards.  If a clock is + * not in sync with the system clock, that could be because the + * system clock isn't keeping accurate time, because the other + * clock isn't keeping accurate time, or both. + * + * Note that host-provided time stamps generally correspond to the + * time when the time-stamping code sees the packet; this could + * be some unknown amount of time after the first or last bit of + * the packet is received by the network adapter, due to batching + * of interrupts for packet arrival, queueing delays, etc.. + */ +#define PCAP_TSTAMP_HOST		0	/* host-provided, unknown characteristics */ +#define PCAP_TSTAMP_HOST_LOWPREC	1	/* host-provided, low precision */ +#define PCAP_TSTAMP_HOST_HIPREC		2	/* host-provided, high precision */ +#define PCAP_TSTAMP_ADAPTER		3	/* device-provided, synced with the system clock */ +#define PCAP_TSTAMP_ADAPTER_UNSYNCED	4	/* device-provided, not synced with the system clock */ + +/* + * Time stamp resolution types. + * Not all systems and interfaces will necessarily support all of these + * resolutions when doing live captures; all of them can be requested + * when reading a savefile. + */ +#define PCAP_TSTAMP_PRECISION_MICRO	0	/* use timestamps with microsecond precision, default */ +#define PCAP_TSTAMP_PRECISION_NANO	1	/* use timestamps with nanosecond precision */ + +PCAP_API pcap_t	*pcap_open_live(const char *, int, int, int, char *); +PCAP_API pcap_t	*pcap_open_dead(int, int); +PCAP_API pcap_t	*pcap_open_dead_with_tstamp_precision(int, int, u_int); +PCAP_API pcap_t	*pcap_open_offline_with_tstamp_precision(const char *, u_int, char *); +PCAP_API pcap_t	*pcap_open_offline(const char *, char *); +#ifdef _WIN32 +  PCAP_API pcap_t  *pcap_hopen_offline_with_tstamp_precision(intptr_t, u_int, char *); +  PCAP_API pcap_t  *pcap_hopen_offline(intptr_t, char *); +  /* +   * If we're building libpcap, these are internal routines in savefile.c, +   * so we must not define them as macros. +   * +   * If we're not building libpcap, given that the version of the C runtime +   * with which libpcap was built might be different from the version +   * of the C runtime with which an application using libpcap was built, +   * and that a FILE structure may differ between the two versions of the +   * C runtime, calls to _fileno() must use the version of _fileno() in +   * the C runtime used to open the FILE *, not the version in the C +   * runtime with which libpcap was built.  (Maybe once the Universal CRT +   * rules the world, this will cease to be a problem.) +   */ +  #ifndef BUILDING_PCAP +    #define pcap_fopen_offline_with_tstamp_precision(f,p,b) \ +	pcap_hopen_offline_with_tstamp_precision(_get_osfhandle(_fileno(f)), p, b) +    #define pcap_fopen_offline(f,b) \ +	pcap_hopen_offline(_get_osfhandle(_fileno(f)), b) +  #endif +#else /*_WIN32*/ +  PCAP_API pcap_t	*pcap_fopen_offline_with_tstamp_precision(FILE *, u_int, char *); +  PCAP_API pcap_t	*pcap_fopen_offline(FILE *, char *); +#endif /*_WIN32*/ + +PCAP_API void	pcap_close(pcap_t *); +PCAP_API int	pcap_loop(pcap_t *, int, pcap_handler, u_char *); +PCAP_API int	pcap_dispatch(pcap_t *, int, pcap_handler, u_char *); +PCAP_API const u_char *pcap_next(pcap_t *, struct pcap_pkthdr *); +PCAP_API int 	pcap_next_ex(pcap_t *, struct pcap_pkthdr **, const u_char **); +PCAP_API void	pcap_breakloop(pcap_t *); +PCAP_API int	pcap_stats(pcap_t *, struct pcap_stat *); +PCAP_API int	pcap_setfilter(pcap_t *, struct bpf_program *); +PCAP_API int 	pcap_setdirection(pcap_t *, pcap_direction_t); +PCAP_API int	pcap_getnonblock(pcap_t *, char *); +PCAP_API int	pcap_setnonblock(pcap_t *, int, char *); +PCAP_API int	pcap_inject(pcap_t *, const void *, size_t); +PCAP_API int	pcap_sendpacket(pcap_t *, const u_char *, int); +PCAP_API const char *pcap_statustostr(int); +PCAP_API const char *pcap_strerror(int); +PCAP_API char	*pcap_geterr(pcap_t *); +PCAP_API void	pcap_perror(pcap_t *, const char *); +PCAP_API int	pcap_compile(pcap_t *, struct bpf_program *, const char *, int, +	    bpf_u_int32); +PCAP_API int	pcap_compile_nopcap(int, int, struct bpf_program *, +	    const char *, int, bpf_u_int32); +PCAP_API void	pcap_freecode(struct bpf_program *); +PCAP_API int	pcap_offline_filter(const struct bpf_program *, +	    const struct pcap_pkthdr *, const u_char *); +PCAP_API int	pcap_datalink(pcap_t *); +PCAP_API int	pcap_datalink_ext(pcap_t *); +PCAP_API int	pcap_list_datalinks(pcap_t *, int **); +PCAP_API int	pcap_set_datalink(pcap_t *, int); +PCAP_API void	pcap_free_datalinks(int *); +PCAP_API int	pcap_datalink_name_to_val(const char *); +PCAP_API const char *pcap_datalink_val_to_name(int); +PCAP_API const char *pcap_datalink_val_to_description(int); +PCAP_API int	pcap_snapshot(pcap_t *); +PCAP_API int	pcap_is_swapped(pcap_t *); +PCAP_API int	pcap_major_version(pcap_t *); +PCAP_API int	pcap_minor_version(pcap_t *); +PCAP_API int	pcap_bufsize(pcap_t *); + +/* XXX */ +PCAP_API FILE	*pcap_file(pcap_t *); +PCAP_API int	pcap_fileno(pcap_t *); + +#ifdef _WIN32 +  PCAP_API int	pcap_wsockinit(void); +#endif + +PCAP_API pcap_dumper_t *pcap_dump_open(pcap_t *, const char *); +PCAP_API pcap_dumper_t *pcap_dump_fopen(pcap_t *, FILE *fp); +PCAP_API pcap_dumper_t *pcap_dump_open_append(pcap_t *, const char *); +PCAP_API FILE	*pcap_dump_file(pcap_dumper_t *); +PCAP_API long	pcap_dump_ftell(pcap_dumper_t *); +PCAP_API int64_t	pcap_dump_ftell64(pcap_dumper_t *); +PCAP_API int	pcap_dump_flush(pcap_dumper_t *); +PCAP_API void	pcap_dump_close(pcap_dumper_t *); +PCAP_API void	pcap_dump(u_char *, const struct pcap_pkthdr *, const u_char *); + +PCAP_API int	pcap_findalldevs(pcap_if_t **, char *); +PCAP_API void	pcap_freealldevs(pcap_if_t *); + +/* + * We return a pointer to the version string, rather than exporting the + * version string directly. + * + * On at least some UNIXes, if you import data from a shared library into + * an program, the data is bound into the program binary, so if the string + * in the version of the library with which the program was linked isn't + * the same as the string in the version of the library with which the + * program is being run, various undesirable things may happen (warnings, + * the string being the one from the version of the library with which the + * program was linked, or even weirder things, such as the string being the + * one from the library but being truncated). + * + * On Windows, the string is constructed at run time. + */ +PCAP_API const char *pcap_lib_version(void); + +/* + * On at least some versions of NetBSD and QNX, we don't want to declare + * bpf_filter() here, as it's also be declared in <net/bpf.h>, with a + * different signature, but, on other BSD-flavored UN*Xes, it's not + * declared in <net/bpf.h>, so we *do* want to declare it here, so it's + * declared when we build pcap-bpf.c. + */ +#if !defined(__NetBSD__) && !defined(__QNX__) +  PCAP_API u_int	bpf_filter(const struct bpf_insn *, const u_char *, u_int, u_int); +#endif +PCAP_API int	bpf_validate(const struct bpf_insn *f, int len); +PCAP_API char	*bpf_image(const struct bpf_insn *, int); +PCAP_API void	bpf_dump(const struct bpf_program *, int); + +#if defined(_WIN32) + +  /* +   * Win32 definitions +   */ + +  /*! +    \brief A queue of raw packets that will be sent to the network with pcap_sendqueue_transmit(). +  */ +  struct pcap_send_queue +  { +	u_int maxlen;	/* Maximum size of the queue, in bytes. This +			   variable contains the size of the buffer field. */ +	u_int len;	/* Current size of the queue, in bytes. */ +	char *buffer;	/* Buffer containing the packets to be sent. */ +  }; + +  typedef struct pcap_send_queue pcap_send_queue; + +  /*! +    \brief This typedef is a support for the pcap_get_airpcap_handle() function +  */ +  #if !defined(AIRPCAP_HANDLE__EAE405F5_0171_9592_B3C2_C19EC426AD34__DEFINED_) +    #define AIRPCAP_HANDLE__EAE405F5_0171_9592_B3C2_C19EC426AD34__DEFINED_ +    typedef struct _AirpcapHandle *PAirpcapHandle; +  #endif + +  PCAP_API int pcap_setbuff(pcap_t *p, int dim); +  PCAP_API int pcap_setmode(pcap_t *p, int mode); +  PCAP_API int pcap_setmintocopy(pcap_t *p, int size); + +  PCAP_API HANDLE pcap_getevent(pcap_t *p); + +  PCAP_API int pcap_oid_get_request(pcap_t *, bpf_u_int32, void *, size_t *); +  PCAP_API int pcap_oid_set_request(pcap_t *, bpf_u_int32, const void *, size_t *); + +  PCAP_API pcap_send_queue* pcap_sendqueue_alloc(u_int memsize); + +  PCAP_API void pcap_sendqueue_destroy(pcap_send_queue* queue); + +  PCAP_API int pcap_sendqueue_queue(pcap_send_queue* queue, const struct pcap_pkthdr *pkt_header, const u_char *pkt_data); + +  PCAP_API u_int pcap_sendqueue_transmit(pcap_t *p, pcap_send_queue* queue, int sync); + +  PCAP_API struct pcap_stat *pcap_stats_ex(pcap_t *p, int *pcap_stat_size); + +  PCAP_API int pcap_setuserbuffer(pcap_t *p, int size); + +  PCAP_API int pcap_live_dump(pcap_t *p, char *filename, int maxsize, int maxpacks); + +  PCAP_API int pcap_live_dump_ended(pcap_t *p, int sync); + +  PCAP_API int pcap_start_oem(char* err_str, int flags); + +  PCAP_API PAirpcapHandle pcap_get_airpcap_handle(pcap_t *p); + +  #define MODE_CAPT 0 +  #define MODE_STAT 1 +  #define MODE_MON 2 + +#elif defined(MSDOS) + +  /* +   * MS-DOS definitions +   */ + +  PCAP_API int  pcap_stats_ex (pcap_t *, struct pcap_stat_ex *); +  PCAP_API void pcap_set_wait (pcap_t *p, void (*yield)(void), int wait); +  PCAP_API u_long pcap_mac_packets (void); + +#else /* UN*X */ + +  /* +   * UN*X definitions +   */ + +  PCAP_API int	pcap_get_selectable_fd(pcap_t *); + +#endif /* _WIN32/MSDOS/UN*X */ + +/* + * Remote capture definitions. + * + * These routines are only present if libpcap has been configured to + * include remote capture support. + */ + +/* + * The maximum buffer size in which address, port, interface names are kept. + * + * In case the adapter name or such is larger than this value, it is truncated. + * This is not used by the user; however it must be aware that an hostname / interface + * name longer than this value will be truncated. + */ +#define PCAP_BUF_SIZE 1024 + +/* + * The type of input source, passed to pcap_open(). + */ +#define PCAP_SRC_FILE		2	/* local savefile */ +#define PCAP_SRC_IFLOCAL	3	/* local network interface */ +#define PCAP_SRC_IFREMOTE	4	/* interface on a remote host, using RPCAP */ + +/* + * The formats allowed by pcap_open() are the following: + * - file://path_and_filename [opens a local file] + * - rpcap://devicename [opens the selected device devices available on the local host, without using the RPCAP protocol] + * - rpcap://host/devicename [opens the selected device available on a remote host] + * - rpcap://host:port/devicename [opens the selected device available on a remote host, using a non-standard port for RPCAP] + * - adaptername [to open a local adapter; kept for compability, but it is strongly discouraged] + * - (NULL) [to open the first local adapter; kept for compability, but it is strongly discouraged] + * + * The formats allowed by the pcap_findalldevs_ex() are the following: + * - file://folder/ [lists all the files in the given folder] + * - rpcap:// [lists all local adapters] + * - rpcap://host:port/ [lists the devices available on a remote host] + * + * Referring to the 'host' and 'port' parameters, they can be either numeric or literal. Since + * IPv6 is fully supported, these are the allowed formats: + * + * - host (literal): e.g. host.foo.bar + * - host (numeric IPv4): e.g. 10.11.12.13 + * - host (numeric IPv4, IPv6 style): e.g. [10.11.12.13] + * - host (numeric IPv6): e.g. [1:2:3::4] + * - port: can be either numeric (e.g. '80') or literal (e.g. 'http') + * + * Here you find some allowed examples: + * - rpcap://host.foo.bar/devicename [everything literal, no port number] + * - rpcap://host.foo.bar:1234/devicename [everything literal, with port number] + * - rpcap://10.11.12.13/devicename [IPv4 numeric, no port number] + * - rpcap://10.11.12.13:1234/devicename [IPv4 numeric, with port number] + * - rpcap://[10.11.12.13]:1234/devicename [IPv4 numeric with IPv6 format, with port number] + * - rpcap://[1:2:3::4]/devicename [IPv6 numeric, no port number] + * - rpcap://[1:2:3::4]:1234/devicename [IPv6 numeric, with port number] + * - rpcap://[1:2:3::4]:http/devicename [IPv6 numeric, with literal port number] + */ + +/* + * URL schemes for capture source. + */ +/* + * This string indicates that the user wants to open a capture from a + * local file. + */ +#define PCAP_SRC_FILE_STRING "file://" +/* + * This string indicates that the user wants to open a capture from a + * network interface.  This string does not necessarily involve the use + * of the RPCAP protocol. If the interface required resides on the local + * host, the RPCAP protocol is not involved and the local functions are used. + */ +#define PCAP_SRC_IF_STRING "rpcap://" + +/* + * Flags to pass to pcap_open(). + */ + +/* + * Specifies whether promiscuous mode is to be used. + */ +#define PCAP_OPENFLAG_PROMISCUOUS		0x00000001 + +/* + * Specifies, for an RPCAP capture, whether the data transfer (in + * case of a remote capture) has to be done with UDP protocol. + * + * If it is '1' if you want a UDP data connection, '0' if you want + * a TCP data connection; control connection is always TCP-based. + * A UDP connection is much lighter, but it does not guarantee that all + * the captured packets arrive to the client workstation. Moreover, + * it could be harmful in case of network congestion. + * This flag is meaningless if the source is not a remote interface. + * In that case, it is simply ignored. + */ +#define PCAP_OPENFLAG_DATATX_UDP		0x00000002 + +/* + * Specifies wheether the remote probe will capture its own generated + * traffic. + * + * In case the remote probe uses the same interface to capture traffic + * and to send data back to the caller, the captured traffic includes + * the RPCAP traffic as well.  If this flag is turned on, the RPCAP + * traffic is excluded from the capture, so that the trace returned + * back to the collector is does not include this traffic. + * + * Has no effect on local interfaces or savefiles. + */ +#define PCAP_OPENFLAG_NOCAPTURE_RPCAP		0x00000004 + +/* + * Specifies whether the local adapter will capture its own generated traffic. + * + * This flag tells the underlying capture driver to drop the packets + * that were sent by itself.  This is useful when building applications + * such as bridges that should ignore the traffic they just sent. + * + * Supported only on Windows. + */ +#define PCAP_OPENFLAG_NOCAPTURE_LOCAL		0x00000008 + +/* + * This flag configures the adapter for maximum responsiveness. + * + * In presence of a large value for nbytes, WinPcap waits for the arrival + * of several packets before copying the data to the user. This guarantees + * a low number of system calls, i.e. lower processor usage, i.e. better + * performance, which is good for applications like sniffers. If the user + * sets the PCAP_OPENFLAG_MAX_RESPONSIVENESS flag, the capture driver will + * copy the packets as soon as the application is ready to receive them. + * This is suggested for real time applications (such as, for example, + * a bridge) that need the best responsiveness. + * + * The equivalent with pcap_create()/pcap_activate() is "immediate mode". + */ +#define PCAP_OPENFLAG_MAX_RESPONSIVENESS	0x00000010 + +/* + * Remote authentication methods. + * These are used in the 'type' member of the pcap_rmtauth structure. + */ + +/* + * NULL authentication. + * + * The 'NULL' authentication has to be equal to 'zero', so that old + * applications can just put every field of struct pcap_rmtauth to zero, + * and it does work. + */ +#define RPCAP_RMTAUTH_NULL 0 +/* + * Username/password authentication. + * + * With this type of authentication, the RPCAP protocol will use the username/ + * password provided to authenticate the user on the remote machine. If the + * authentication is successful (and the user has the right to open network + * devices) the RPCAP connection will continue; otherwise it will be dropped. + * + * *******NOTE********: the username and password are sent over the network + * to the capture server *IN CLEAR TEXT*.  Don't use this on a network + * that you don't completely control!  (And be *really* careful in your + * definition of "completely"!) + */ +#define RPCAP_RMTAUTH_PWD 1 + +/* + * This structure keeps the information needed to autheticate the user + * on a remote machine. + * + * The remote machine can either grant or refuse the access according + * to the information provided. + * In case the NULL authentication is required, both 'username' and + * 'password' can be NULL pointers. + * + * This structure is meaningless if the source is not a remote interface; + * in that case, the functions which requires such a structure can accept + * a NULL pointer as well. + */ +struct pcap_rmtauth +{ +	/* +	 * \brief Type of the authentication required. +	 * +	 * In order to provide maximum flexibility, we can support different types +	 * of authentication based on the value of this 'type' variable. The currently +	 * supported authentication methods are defined into the +	 * \link remote_auth_methods Remote Authentication Methods Section\endlink. +	 */ +	int type; +	/* +	 * \brief Zero-terminated string containing the username that has to be +	 * used on the remote machine for authentication. +	 * +	 * This field is meaningless in case of the RPCAP_RMTAUTH_NULL authentication +	 * and it can be NULL. +	 */ +	char *username; +	/* +	 * \brief Zero-terminated string containing the password that has to be +	 * used on the remote machine for authentication. +	 * +	 * This field is meaningless in case of the RPCAP_RMTAUTH_NULL authentication +	 * and it can be NULL. +	 */ +	char *password; +}; + +/* + * This routine can open a savefile, a local device, or a device on + * a remote machine running an RPCAP server. + * + * For opening a savefile, the pcap_open_offline routines can be used, + * and will work just as well; code using them will work on more + * platforms than code using pcap_open() to open savefiles. + * + * For opening a local device, pcap_open_live() can be used; it supports + * most of the capabilities that pcap_open() supports, and code using it + * will work on more platforms than code using pcap_open().  pcap_create() + * and pcap_activate() can also be used; they support all capabilities + * that pcap_open() supports, except for the Windows-only + * PCAP_OPENFLAG_NOCAPTURE_LOCAL, and support additional capabilities. + * + * For opening a remote capture, pcap_open() is currently the only + * API available. + */ +PCAP_API pcap_t	*pcap_open(const char *source, int snaplen, int flags, +	    int read_timeout, struct pcap_rmtauth *auth, char *errbuf); +PCAP_API int	pcap_createsrcstr(char *source, int type, const char *host, +	    const char *port, const char *name, char *errbuf); +PCAP_API int	pcap_parsesrcstr(const char *source, int *type, char *host, +	    char *port, char *name, char *errbuf); + +/* + * This routine can scan a directory for savefiles, list local capture + * devices, or list capture devices on a remote machine running an RPCAP + * server. + * + * For scanning for savefiles, it can be used on both UN*X systems and + * Windows systems; for each directory entry it sees, it tries to open + * the file as a savefile using pcap_open_offline(), and only includes + * it in the list of files if the open succeeds, so it filters out + * files for which the user doesn't have read permission, as well as + * files that aren't valid savefiles readable by libpcap. + * + * For listing local capture devices, it's just a wrapper around + * pcap_findalldevs(); code using pcap_findalldevs() will work on more + * platforms than code using pcap_findalldevs_ex(). + * + * For listing remote capture devices, pcap_findalldevs_ex() is currently + * the only API available. + */ +PCAP_API int	pcap_findalldevs_ex(char *source, struct pcap_rmtauth *auth, +	    pcap_if_t **alldevs, char *errbuf); + +/* + * Sampling methods. + * + * These allow pcap_loop(), pcap_dispatch(), pcap_next(), and pcap_next_ex() + * to see only a sample of packets, rather than all packets. + * + * Currently, they work only on Windows local captures. + */ + +/* + * Specifies that no sampling is to be done on the current capture. + * + * In this case, no sampling algorithms are applied to the current capture. + */ +#define PCAP_SAMP_NOSAMP	0 + +/* + * Specifies that only 1 out of N packets must be returned to the user. + * + * In this case, the 'value' field of the 'pcap_samp' structure indicates the + * number of packets (minus 1) that must be discarded before one packet got + * accepted. + * In other words, if 'value = 10', the first packet is returned to the + * caller, while the following 9 are discarded. + */ +#define PCAP_SAMP_1_EVERY_N	1 + +/* + * Specifies that we have to return 1 packet every N milliseconds. + * + * In this case, the 'value' field of the 'pcap_samp' structure indicates + * the 'waiting time' in milliseconds before one packet got accepted. + * In other words, if 'value = 10', the first packet is returned to the + * caller; the next returned one will be the first packet that arrives + * when 10ms have elapsed. + */ +#define PCAP_SAMP_FIRST_AFTER_N_MS 2 + +/* + * This structure defines the information related to sampling. + * + * In case the sampling is requested, the capturing device should read + * only a subset of the packets coming from the source. The returned packets + * depend on the sampling parameters. + * + * WARNING: The sampling process is applied *after* the filtering process. + * In other words, packets are filtered first, then the sampling process + * selects a subset of the 'filtered' packets and it returns them to the + * caller. + */ +struct pcap_samp +{ +	/* +	 * Method used for sampling; see above. +	 */ +	int method; + +	/* +	 * This value depends on the sampling method defined. +	 * For its meaning, see above. +	 */ +	int value; +}; + +/* + * New functions. + */ +PCAP_API struct pcap_samp *pcap_setsampling(pcap_t *p); + +/* + * RPCAP active mode. + */ + +/* Maximum length of an host name (needed for the RPCAP active mode) */ +#define RPCAP_HOSTLIST_SIZE 1024 + +/* + * Some minor differences between UN*X sockets and and Winsock sockets. + */ +#ifndef _WIN32 +  /*! +   * \brief In Winsock, a socket handle is of type SOCKET; in UN*X, it's +   * a file descriptor, and therefore a signed integer. +   * We define SOCKET to be a signed integer on UN*X, so that it can +   * be used on both platforms. +   */ +  #define SOCKET int + +  /*! +   * \brief In Winsock, the error return if socket() fails is INVALID_SOCKET; +   * in UN*X, it's -1. +   * We define INVALID_SOCKET to be -1 on UN*X, so that it can be used on +   * both platforms. +   */ +  #define INVALID_SOCKET -1 +#endif + +PCAP_API SOCKET	pcap_remoteact_accept(const char *address, const char *port, +	    const char *hostlist, char *connectinghost, +	    struct pcap_rmtauth *auth, char *errbuf); +PCAP_API int	pcap_remoteact_list(char *hostlist, char sep, int size, +	    char *errbuf); +PCAP_API int	pcap_remoteact_close(const char *host, char *errbuf); +PCAP_API void	pcap_remoteact_cleanup(void); + +#ifdef __cplusplus +} +#endif + +#endif /* lib_pcap_pcap_h */ diff --git a/src/frontend/qt_sdl/pcap/sll.h b/src/frontend/qt_sdl/pcap/sll.h new file mode 100644 index 0000000..c4d0886 --- /dev/null +++ b/src/frontend/qt_sdl/pcap/sll.h @@ -0,0 +1,131 @@ +/*- + * Copyright (c) 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997 + *	The Regents of the University of California.  All rights reserved. + * + * This code is derived from the Stanford/CMU enet packet filter, + * (net/enet.c) distributed as part of 4.3BSD, and code contributed + * to Berkeley by Steven McCanne and Van Jacobson both of Lawrence + * Berkeley Laboratory. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + *    must display the following acknowledgement: + *      This product includes software developed by the University of + *      California, Berkeley and its contributors. + * 4. Neither the name of the University nor the names of its contributors + *    may be used to endorse or promote products derived from this software + *    without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +/* + * For captures on Linux cooked sockets, we construct a fake header + * that includes: + * + *	a 2-byte "packet type" which is one of: + * + *		LINUX_SLL_HOST		packet was sent to us + *		LINUX_SLL_BROADCAST	packet was broadcast + *		LINUX_SLL_MULTICAST	packet was multicast + *		LINUX_SLL_OTHERHOST	packet was sent to somebody else + *		LINUX_SLL_OUTGOING	packet was sent *by* us; + * + *	a 2-byte Ethernet protocol field; + * + *	a 2-byte link-layer type; + * + *	a 2-byte link-layer address length; + * + *	an 8-byte source link-layer address, whose actual length is + *	specified by the previous value. + * + * All fields except for the link-layer address are in network byte order. + * + * DO NOT change the layout of this structure, or change any of the + * LINUX_SLL_ values below.  If you must change the link-layer header + * for a "cooked" Linux capture, introduce a new DLT_ type (ask + * "tcpdump-workers@lists.tcpdump.org" for one, so that you don't give it + * a value that collides with a value already being used), and use the + * new header in captures of that type, so that programs that can + * handle DLT_LINUX_SLL captures will continue to handle them correctly + * without any change, and so that capture files with different headers + * can be told apart and programs that read them can dissect the + * packets in them. + */ + +#ifndef lib_pcap_sll_h +#define lib_pcap_sll_h + +/* + * A DLT_LINUX_SLL fake link-layer header. + */ +#define SLL_HDR_LEN	16		/* total header length */ +#define SLL_ADDRLEN	8		/* length of address field */ + +#include <pcap/pcap-inttypes.h> + +struct sll_header { +	uint16_t sll_pkttype;		/* packet type */ +	uint16_t sll_hatype;		/* link-layer address type */ +	uint16_t sll_halen;		/* link-layer address length */ +	uint8_t sll_addr[SLL_ADDRLEN];	/* link-layer address */ +	uint16_t sll_protocol;		/* protocol */ +}; + +/* + * The LINUX_SLL_ values for "sll_pkttype"; these correspond to the + * PACKET_ values on Linux, but are defined here so that they're + * available even on systems other than Linux, and so that they + * don't change even if the PACKET_ values change. + */ +#define LINUX_SLL_HOST		0 +#define LINUX_SLL_BROADCAST	1 +#define LINUX_SLL_MULTICAST	2 +#define LINUX_SLL_OTHERHOST	3 +#define LINUX_SLL_OUTGOING	4 + +/* + * The LINUX_SLL_ values for "sll_protocol"; these correspond to the + * ETH_P_ values on Linux, but are defined here so that they're + * available even on systems other than Linux.  We assume, for now, + * that the ETH_P_ values won't change in Linux; if they do, then: + * + *	if we don't translate them in "pcap-linux.c", capture files + *	won't necessarily be readable if captured on a system that + *	defines ETH_P_ values that don't match these values; + * + *	if we do translate them in "pcap-linux.c", that makes life + *	unpleasant for the BPF code generator, as the values you test + *	for in the kernel aren't the values that you test for when + *	reading a capture file, so the fixup code run on BPF programs + *	handed to the kernel ends up having to do more work. + * + * Add other values here as necessary, for handling packet types that + * might show up on non-Ethernet, non-802.x networks.  (Not all the ones + * in the Linux "if_ether.h" will, I suspect, actually show up in + * captures.) + */ +#define LINUX_SLL_P_802_3	0x0001	/* Novell 802.3 frames without 802.2 LLC header */ +#define LINUX_SLL_P_802_2	0x0004	/* 802.2 frames (not D/I/X Ethernet) */ +#define LINUX_SLL_P_CAN		0x000C	/* CAN frames, with SocketCAN pseudo-headers */ +#define LINUX_SLL_P_CANFD	0x000D	/* CAN FD frames, with SocketCAN pseudo-headers */ + +#endif diff --git a/src/frontend/qt_sdl/pcap/usb.h b/src/frontend/qt_sdl/pcap/usb.h new file mode 100644 index 0000000..e485ec8 --- /dev/null +++ b/src/frontend/qt_sdl/pcap/usb.h @@ -0,0 +1,143 @@ +/* + * Copyright (c) 2006 Paolo Abeni (Italy) + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. The name of the author may not be used to endorse or promote + * products derived from this software without specific prior written + * permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + * Basic USB data struct + * By Paolo Abeni <paolo.abeni@email.it> + */ + +#ifndef lib_pcap_usb_h +#define lib_pcap_usb_h + +#include <pcap/pcap-inttypes.h> + +/* + * possible transfer mode + */ +#define URB_TRANSFER_IN   0x80 +#define URB_ISOCHRONOUS   0x0 +#define URB_INTERRUPT     0x1 +#define URB_CONTROL       0x2 +#define URB_BULK          0x3 + +/* + * possible event type + */ +#define URB_SUBMIT        'S' +#define URB_COMPLETE      'C' +#define URB_ERROR         'E' + +/* + * USB setup header as defined in USB specification. + * Appears at the front of each Control S-type packet in DLT_USB captures. + */ +typedef struct _usb_setup { +	uint8_t bmRequestType; +	uint8_t bRequest; +	uint16_t wValue; +	uint16_t wIndex; +	uint16_t wLength; +} pcap_usb_setup; + +/* + * Information from the URB for Isochronous transfers. + */ +typedef struct _iso_rec { +	int32_t	error_count; +	int32_t	numdesc; +} iso_rec; + +/* + * Header prepended by linux kernel to each event. + * Appears at the front of each packet in DLT_USB_LINUX captures. + */ +typedef struct _usb_header { +	uint64_t id; +	uint8_t event_type; +	uint8_t transfer_type; +	uint8_t endpoint_number; +	uint8_t device_address; +	uint16_t bus_id; +	char setup_flag;/*if !=0 the urb setup header is not present*/ +	char data_flag; /*if !=0 no urb data is present*/ +	int64_t ts_sec; +	int32_t ts_usec; +	int32_t status; +	uint32_t urb_len; +	uint32_t data_len; /* amount of urb data really present in this event*/ +	pcap_usb_setup setup; +} pcap_usb_header; + +/* + * Header prepended by linux kernel to each event for the 2.6.31 + * and later kernels; for the 2.6.21 through 2.6.30 kernels, the + * "iso_rec" information, and the fields starting with "interval" + * are zeroed-out padding fields. + * + * Appears at the front of each packet in DLT_USB_LINUX_MMAPPED captures. + */ +typedef struct _usb_header_mmapped { +	uint64_t id; +	uint8_t event_type; +	uint8_t transfer_type; +	uint8_t endpoint_number; +	uint8_t device_address; +	uint16_t bus_id; +	char setup_flag;/*if !=0 the urb setup header is not present*/ +	char data_flag; /*if !=0 no urb data is present*/ +	int64_t ts_sec; +	int32_t ts_usec; +	int32_t status; +	uint32_t urb_len; +	uint32_t data_len; /* amount of urb data really present in this event*/ +	union { +		pcap_usb_setup setup; +		iso_rec iso; +	} s; +	int32_t	interval;	/* for Interrupt and Isochronous events */ +	int32_t start_frame;	/* for Isochronous events */ +	uint32_t xfer_flags;	/* copy of URB's transfer flags */ +	uint32_t ndesc;	/* number of isochronous descriptors */ +} pcap_usb_header_mmapped; + +/* + * Isochronous descriptors; for isochronous transfers there might be + * one or more of these at the beginning of the packet data.  The + * number of descriptors is given by the "ndesc" field in the header; + * as indicated, in older kernels that don't put the descriptors at + * the beginning of the packet, that field is zeroed out, so that field + * can be trusted even in captures from older kernels. + */ +typedef struct _usb_isodesc { +	int32_t		status; +	uint32_t	offset; +	uint32_t	len; +	uint8_t	pad[4]; +} usb_isodesc; + +#endif diff --git a/src/frontend/qt_sdl/pcap/vlan.h b/src/frontend/qt_sdl/pcap/vlan.h new file mode 100644 index 0000000..b29dd73 --- /dev/null +++ b/src/frontend/qt_sdl/pcap/vlan.h @@ -0,0 +1,46 @@ +/*- + * Copyright (c) 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997 + *	The Regents of the University of California.  All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + *    must display the following acknowledgement: + *      This product includes software developed by the University of + *      California, Berkeley and its contributors. + * 4. Neither the name of the University nor the names of its contributors + *    may be used to endorse or promote products derived from this software + *    without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +#ifndef lib_pcap_vlan_h +#define lib_pcap_vlan_h + +#include <pcap/pcap-inttypes.h> + +struct vlan_tag { +	uint16_t	vlan_tpid;		/* ETH_P_8021Q */ +	uint16_t	vlan_tci;		/* VLAN TCI */ +}; + +#define VLAN_TAG_LEN	4 + +#endif |