summaryrefslogblamecommitdiffstats
path: root/src/core/hle/service/service.cpp
blob: 44c7c8627c4433200e3bb35da8cc44be147b90c6 (plain) (tree)
1
2
3
4
5
6
7
8
9






                                          
                         
                                     
                                 
 



                                                


                                                                                                    















































                                                                           


                                                                                                    
                              
 

       
           

     
            

     





                                                                                               



                                                       
                                 






                                                       
                                     







                                                                                         
                                


                                                                                 
 
                             
                                                        


                                   
         











                                                                                             





                                                                                


                  


                          
     
     

  


                                                                                                    


                             
                                   
                                   
                                               




                           
                                            



 
// Copyright 2014 Citra Emulator Project
// Licensed under GPLv2
// Refer to the license.txt file included.

#include "common/common.h"
#include "common/log.h"

#include "core/hle/hle.h"
#include "core/hle/service/service.h"
#include "core/hle/service/apt.h"

namespace Service {

Manager* g_manager = NULL;  ///< Service manager

////////////////////////////////////////////////////////////////////////////////////////////////////
// Service Manager class

Manager::Manager() {
}

Manager::~Manager() {
    for(Interface* service : m_services) {
        DeleteService(service->GetPortName());
    }
}

/// Add a service to the manager (does not create it though)
void Manager::AddService(Interface* service) {
    int index = m_services.size();
    u32 new_uid = GetUIDFromIndex(index);

    m_services.push_back(service);

    m_port_map[service->GetPortName()] = new_uid;
    service->m_uid = new_uid;
}

/// Removes a service from the manager, also frees memory
void Manager::DeleteService(std::string port_name) {
    auto service = FetchFromPortName(port_name);

    m_services.erase(m_services.begin() + GetIndexFromUID(service->m_uid));
    m_port_map.erase(port_name);

    delete service;
}

/// Get a Service Interface from its UID
Interface* Manager::FetchFromUID(u32 uid) {
    int index = GetIndexFromUID(uid);
    if (index < (int)m_services.size()) {
        return m_services[index];
    }
    return NULL;
}

/// Get a Service Interface from its port
Interface* Manager::FetchFromPortName(std::string port_name) {
    auto itr = m_port_map.find(port_name);
    if (itr == m_port_map.end()) {
        return NULL;
    }
    return FetchFromUID(itr->second);
}

////////////////////////////////////////////////////////////////////////////////////////////////////
// Interface to "SRV" service

class SRV : public Interface {

public:

    SRV() {
    }

    ~SRV() {
    }

    enum {
        CMD_OFFSET              = 0x80,
        CMD_HEADER_INIT         = 0x10002,  ///< Command header to initialize SRV service
        CMD_HEADER_GET_HANDLE   = 0x50100,  ///< Command header to get handle of other services
    };

    /**
     * Gets the string name used by CTROS for a service
     * @return String name of service
     */
    std::string GetName() const {
        return "ServiceManager";
    }

    /**
     * Gets the string name used by CTROS for a service
     * @return Port name of service
     */
    std::string GetPortName() const {
        return "srv:";
    }

    /**
     * Called when svcSendSyncRequest is called, loads command buffer and executes comand
     * @return Return result of svcSendSyncRequest passed back to user app
     */
    Syscall::Result Sync() {
        Syscall::Result res = 0;
        u32* cmd_buff = (u32*)HLE::GetPointer(HLE::CMD_BUFFER_ADDR + CMD_OFFSET);

        switch (cmd_buff[0]) {

        case CMD_HEADER_INIT:
            NOTICE_LOG(OSHLE, "SRV::Sync - Initialize");
            break;

        case CMD_HEADER_GET_HANDLE:
        {
            const char* port_name = (const char*)&cmd_buff[1];
            Interface* service = g_manager->FetchFromPortName(port_name);

            NOTICE_LOG(OSHLE, "SRV::Sync - GetHandle - port: %s, handle: 0x%08X", port_name, 
                service->GetUID());

            if (NULL != service) {
                cmd_buff[3] = service->GetUID();
            } else {
                ERROR_LOG(OSHLE, "Service %s does not exist", port_name);
                res = -1;
            }
            break;
        }

        default:
            ERROR_LOG(OSHLE, "SRV::Sync - Unknown command 0x%08X", cmd_buff[0]);
            res = -1;
            break;
        }

        cmd_buff[1] = res;

        return res;
    }
     
};

////////////////////////////////////////////////////////////////////////////////////////////////////
// Module interface

/// Initialize ServiceManager
void Init() {
    g_manager = new Manager;
    g_manager->AddService(new SRV);
    g_manager->AddService(new APT);
    NOTICE_LOG(HLE, "Services initialized OK");
}

/// Shutdown ServiceManager
void Shutdown() {
    delete g_manager;
    NOTICE_LOG(HLE, "Services shutdown OK");
}


}