304 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			304 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 *  QCMA: Cross-platform content manager assistant for the PS Vita
 | 
						|
 *
 | 
						|
 *  Copyright (C) 2013  Codestation
 | 
						|
 *
 | 
						|
 *  This program is free software: you can redistribute it and/or modify
 | 
						|
 *  it under the terms of the GNU General Public License as published by
 | 
						|
 *  the Free Software Foundation, either version 3 of the License, or
 | 
						|
 *  (at your option) any later version.
 | 
						|
 *
 | 
						|
 *  This program is distributed in the hope that it will be useful,
 | 
						|
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
 *  GNU General Public License for more details.
 | 
						|
 *
 | 
						|
 *  You should have received a copy of the GNU General Public License
 | 
						|
 *  along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 */
 | 
						|
 | 
						|
#include "cmaclient.h"
 | 
						|
#include "capability.h"
 | 
						|
#include "cmaevent.h"
 | 
						|
#include "cmautils.h"
 | 
						|
 | 
						|
#include <QDateTime>
 | 
						|
#include <QDebug>
 | 
						|
#include <QDir>
 | 
						|
#include <QImage>
 | 
						|
#include <QTextStream>
 | 
						|
#include <QTime>
 | 
						|
#include <QSettings>
 | 
						|
#include <QUrl>
 | 
						|
 | 
						|
#include <random>
 | 
						|
 | 
						|
QMutex CmaClient::mutex;
 | 
						|
QMutex CmaClient::runner;
 | 
						|
QWaitCondition CmaClient::usbcondition;
 | 
						|
QMutex CmaClient::usbwait;
 | 
						|
QSemaphore CmaClient::sema;
 | 
						|
 | 
						|
QString CmaClient::tempOnlineId = QString();
 | 
						|
 | 
						|
bool CmaClient::is_active = false;
 | 
						|
bool CmaClient::in_progress = false;
 | 
						|
 | 
						|
CmaClient *CmaClient::this_object = NULL;
 | 
						|
 | 
						|
CmaClient::CmaClient(Database *db, QObject *obj_parent) :
 | 
						|
    QObject(obj_parent), m_db(db), m_broadcast(NULL)
 | 
						|
{
 | 
						|
    this_object = this;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
CmaClient::CmaClient(Database *db, CmaBroadcast *broadcast, QObject *obj_parent) :
 | 
						|
    QObject(obj_parent), m_db(db), m_broadcast(broadcast)
 | 
						|
{
 | 
						|
    this_object = this;
 | 
						|
}
 | 
						|
 | 
						|
void CmaClient::connectUsb()
 | 
						|
{
 | 
						|
    vita_device_t *vita;
 | 
						|
 | 
						|
    qDebug("Starting usb_thread: 0x%016" PRIxPTR, (uintptr_t)QThread::currentThreadId());
 | 
						|
 | 
						|
    setActive(true);
 | 
						|
    usbwait.lock();
 | 
						|
 | 
						|
    do {
 | 
						|
        if((vita = VitaMTP_Get_First_USB_Vita()) != NULL) {
 | 
						|
            qDebug("Starting new USB connection");
 | 
						|
            processNewConnection(vita);
 | 
						|
        } else {
 | 
						|
            //TODO: replace this with an event-driven setup
 | 
						|
            usbcondition.wait(&usbwait, 2000);
 | 
						|
            mutex.lock();
 | 
						|
            if(in_progress) {
 | 
						|
                sema.acquire();
 | 
						|
            }
 | 
						|
            mutex.unlock();
 | 
						|
        }
 | 
						|
    } while(isActive());
 | 
						|
 | 
						|
    usbwait.unlock();
 | 
						|
    qDebug("Finishing usb_thread");
 | 
						|
    emit finished();
 | 
						|
}
 | 
						|
 | 
						|
void CmaClient::connectWireless()
 | 
						|
{
 | 
						|
    vita_device_t *vita;
 | 
						|
    wireless_host_info_t host = {NULL, NULL, NULL, QCMA_REQUEST_PORT};
 | 
						|
    typedef CmaClient CC;
 | 
						|
 | 
						|
    qDebug("Starting wireless_thread: 0x%016" PRIxPTR, (uintptr_t)QThread::currentThreadId());
 | 
						|
 | 
						|
    setActive(true);
 | 
						|
 | 
						|
    do {
 | 
						|
        qDebug("Waiting for wireless connection");
 | 
						|
        if((vita = VitaMTP_Get_First_Wireless_Vita(&host, 0, CC::deviceRegistered, CC::generatePin, CC::registrationComplete)) != NULL) {
 | 
						|
            qDebug("Starting new wireless connection");
 | 
						|
            m_broadcast->setUnavailable();
 | 
						|
            processNewConnection(vita);
 | 
						|
            m_broadcast->setAvailable();
 | 
						|
        } else {
 | 
						|
            mutex.lock();
 | 
						|
            if(in_progress) {
 | 
						|
                sema.acquire();
 | 
						|
            }
 | 
						|
            mutex.unlock();
 | 
						|
 | 
						|
            // if is active then something happened while setting the socket, wait a little and try again
 | 
						|
            if(isActive()) {
 | 
						|
                qDebug("Error getting wireless connection");
 | 
						|
                Sleeper::sleep(2000);
 | 
						|
            } else {
 | 
						|
                qDebug("Wireless connection cancelled by the user");
 | 
						|
            }
 | 
						|
        }
 | 
						|
    } while(isActive());
 | 
						|
 | 
						|
    qDebug("Finishing wireless_thread");
 | 
						|
    emit finished();
 | 
						|
}
 | 
						|
 | 
						|
void CmaClient::processNewConnection(vita_device_t *device)
 | 
						|
{
 | 
						|
    QMutexLocker locker(&mutex);
 | 
						|
    in_progress = true;
 | 
						|
 | 
						|
    QTextStream(stdout) << "Vita connected, id: " << VitaMTP_Get_Identification(device) << Qt::endl;
 | 
						|
    DeviceCapability vita_info;
 | 
						|
 | 
						|
    if(!vita_info.exchangeInfo(device)) {
 | 
						|
        qCritical("Error while exchanging info with the vita");
 | 
						|
        if(VitaMTP_Get_Device_Type(device) == VitaDeviceUSB) {
 | 
						|
            // reset the device
 | 
						|
            VitaMTP_USB_Reset(device);
 | 
						|
        }
 | 
						|
    } else {
 | 
						|
        QSettings settings;
 | 
						|
 | 
						|
        // Conection successful, inform the user
 | 
						|
        if(vita_info.getOnlineId() != NULL) {
 | 
						|
            settings.setValue("lastOnlineId", vita_info.getOnlineId());
 | 
						|
            emit deviceConnected(QString(tr("Connected to %1 (PS Vita)")).arg(vita_info.getOnlineId()));
 | 
						|
        } else {
 | 
						|
            QString onlineId = settings.value("lastOnlineId", "default").toString();
 | 
						|
            emit deviceConnected(QString(tr("Connected to %1 (PS Vita)")).arg(onlineId));
 | 
						|
        }
 | 
						|
 | 
						|
        enterEventLoop(device);
 | 
						|
    }
 | 
						|
 | 
						|
    VitaMTP_SendHostStatus(device, VITA_HOST_STATUS_EndConnection);
 | 
						|
    qDebug("Releasing device...");
 | 
						|
    VitaMTP_Release_Device(device);
 | 
						|
 | 
						|
    emit deviceDisconnected();
 | 
						|
 | 
						|
    in_progress = false;
 | 
						|
    sema.release();
 | 
						|
}
 | 
						|
 | 
						|
void CmaClient::registrationComplete()
 | 
						|
{
 | 
						|
    QSettings settings;
 | 
						|
    qDebug("Registration completed");
 | 
						|
    settings.setValue("lastOnlineId", tempOnlineId);
 | 
						|
    emit this_object->pinComplete();
 | 
						|
}
 | 
						|
 | 
						|
int CmaClient::deviceRegistered(const char *deviceid)
 | 
						|
{
 | 
						|
    qDebug("Got connection request from %s", deviceid);
 | 
						|
    // TODO: check the device to see if is already registered
 | 
						|
    return 1;
 | 
						|
}
 | 
						|
 | 
						|
int CmaClient::generatePin(wireless_vita_info_t *info, int *p_err)
 | 
						|
{
 | 
						|
    // save the device name in a temporal variable, just in case the pin is rejected
 | 
						|
    tempOnlineId = QString(info->name);
 | 
						|
    qDebug("Registration request from %s (MAC: %s)", info->name, info->mac_addr);
 | 
						|
 | 
						|
    QString staticPin = QSettings().value("staticPin").toString();
 | 
						|
 | 
						|
    static std::mt19937 rng(std::random_device{}());
 | 
						|
 | 
						|
    std::uniform_int_distribution<int> dist(0, 99999999);
 | 
						|
 | 
						|
    int pin;
 | 
						|
 | 
						|
    if(!staticPin.isNull() && staticPin.length() == 8) {
 | 
						|
        bool ok;
 | 
						|
        pin = staticPin.toInt(&ok);
 | 
						|
 | 
						|
        if(!ok) {
 | 
						|
            pin = dist(rng);
 | 
						|
        }
 | 
						|
    } else {
 | 
						|
        pin = dist(rng);
 | 
						|
    }
 | 
						|
 | 
						|
    QTextStream out(stdout);
 | 
						|
    out << "Your registration PIN for " << info->name << " is: ";
 | 
						|
    out.setFieldWidth(8);
 | 
						|
    out.setPadChar('0');
 | 
						|
    out << pin << Qt::endl;
 | 
						|
 | 
						|
    qDebug("PIN: %08i", pin);
 | 
						|
 | 
						|
    *p_err = 0;
 | 
						|
    emit this_object->receivedPin(info->name, pin);
 | 
						|
    return pin;
 | 
						|
}
 | 
						|
 | 
						|
void CmaClient::enterEventLoop(vita_device_t *device)
 | 
						|
{
 | 
						|
    vita_event_t obj_event;
 | 
						|
 | 
						|
    qDebug("Starting event loop");
 | 
						|
 | 
						|
    CmaEvent eventLoop(m_db, device);
 | 
						|
    QThread obj_thread;
 | 
						|
    obj_thread.setObjectName("event_thread");
 | 
						|
 | 
						|
    eventLoop.moveToThread(&obj_thread);
 | 
						|
    connect(&obj_thread, SIGNAL(started()), &eventLoop, SLOT(process()));
 | 
						|
    connect(&eventLoop, SIGNAL(refreshDatabase()), this, SIGNAL(refreshDatabase()), Qt::DirectConnection);
 | 
						|
    connect(&eventLoop, SIGNAL(finishedEventLoop()), &obj_thread, SLOT(quit()), Qt::DirectConnection);
 | 
						|
    connect(&eventLoop, SIGNAL(messageSent(QString)), this, SIGNAL(messageSent(QString)), Qt::DirectConnection);
 | 
						|
    obj_thread.start();
 | 
						|
 | 
						|
    while(isActive()) {
 | 
						|
        if(VitaMTP_Read_Event(device, &obj_event) < 0) {
 | 
						|
            qWarning("Error reading event from Vita.");
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        // do not create a event for this since there aren't more events to read
 | 
						|
        if(obj_event.Code == PTP_EC_VITA_RequestTerminate) {
 | 
						|
            qDebug("Terminating event thread");
 | 
						|
            break;
 | 
						|
 | 
						|
            // this one shuold be processed inmediately
 | 
						|
        } else if(obj_event.Code == PTP_EC_VITA_RequestCancelTask) {
 | 
						|
            eventLoop.vitaEventCancelTask(&obj_event, obj_event.Param1);
 | 
						|
            qDebug("Ended event, code: 0x%x, id: %d", obj_event.Code, obj_event.Param1);
 | 
						|
            continue;
 | 
						|
        }
 | 
						|
 | 
						|
        // the events are processed synchronously except for cancel/terminate
 | 
						|
        qDebug("Sending new event");
 | 
						|
        eventLoop.setEvent(obj_event);
 | 
						|
    }
 | 
						|
 | 
						|
    eventLoop.stop();
 | 
						|
    obj_thread.wait();
 | 
						|
    qDebug("Finishing event loop");
 | 
						|
}
 | 
						|
 | 
						|
int CmaClient::stop()
 | 
						|
{
 | 
						|
    QTextStream(stdout) << "Stopping Qcma" << Qt::endl;
 | 
						|
 | 
						|
    if(!isActive()) {
 | 
						|
        return -1;
 | 
						|
    }
 | 
						|
    CmaClient::setActive(false);
 | 
						|
    qDebug("Cancelling wireless server thread");
 | 
						|
    VitaMTP_Cancel_Get_Wireless_Vita();
 | 
						|
    usbcondition.wakeAll();
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
bool CmaClient::isActive()
 | 
						|
{
 | 
						|
    QMutexLocker locker(&runner);
 | 
						|
    return is_active;
 | 
						|
}
 | 
						|
 | 
						|
void CmaClient::setActive(bool state)
 | 
						|
{
 | 
						|
    QMutexLocker locker(&runner);
 | 
						|
    is_active = state;
 | 
						|
}
 | 
						|
 | 
						|
bool CmaClient::isRunning()
 | 
						|
{
 | 
						|
    bool ret;
 | 
						|
    if(mutex.tryLock()) {
 | 
						|
        ret = in_progress;
 | 
						|
        mutex.unlock();
 | 
						|
    } else {
 | 
						|
        ret = true;
 | 
						|
    }
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 |