对于Sigaction的交替

时间:2013-11-15 09:58:20

标签: c++ linux signals

我在使用此类的C ++中使用sigaction时遇到问题:

class SerialHandler{
                private:
                                 ...

                        /* Action Handling */
                        struct sigaction receive_handler;
                        /* Called using the receive_handler*/
                        static void receive_function(int status);
                        /* Get byte from the Serial Port */
                        unsigned char receive_byte();
                        /* Buffer for Singalinterrupt on Reciving */
                        std::vector<char> in_char_buffer;
                                 ...
};

我真的需要使用中断,并且确实需要在中断/ Sigaction上使用成员函数(receive_function),因为我需要访问成员变量(vector in_char_buffer)。问题:您不能使用sigaction使用成员函数,因为您必须将正常函数传递给sigaction.sa_handler。

正如您所看到的,使用static函数也不是替代方法,因为您将无法访问成员变量。包装也无济于事。您可以访问新对象,但不能访问特定对象。

有没有可以解决这个问题的替代方案?

1 个答案:

答案 0 :(得分:1)

这样的事情怎么样:

#include <signal.h>
#include <string.h>

#include <vector>

using namespace std;

class SerialHandler;

// A Singleton class
class SerialHandlerManager
{
private:
    SerialHandlerManager() {};

public:
    static SerialHandlerManager* get_instance();
    void register_handler( int signum, int fd, SerialHandler * handler );

private:
    static void dispatch_signal( int signum, siginfo_t * info, void * context );

private:
    static SerialHandlerManager* the_manager;
    vector<SerialHandler *> handler_objects;
};

class SerialHandler
{
private:
    friend class SerialHandlerManager;
    void install_signal_handler( int signum )
    {
        (SerialHandlerManager::get_instance())->register_handler( signum, fd, this );
    }

    void receive_function( int status );

    int fd;     // file descriptor returned from opening the serial port
};

SerialHandlerManager* SerialHandlerManager::the_manager;

SerialHandlerManager* SerialHandlerManager::get_instance()
{
    if( !the_manager )
        the_manager = new SerialHandlerManager;
    return the_manager;
}

void SerialHandlerManager::register_handler( int signum, int fd, SerialHandler * handler )
{
    struct sigaction act;

    memset( &act, 0, sizeof(struct sigaction));

    act.sa_flags = SA_SIGINFO;
    act.sa_sigaction = SerialHandlerManager::dispatch_signal;

    handler_objects[fd] = handler;
    sigaction( signum, &act, NULL );

}

void SerialHandlerManager::dispatch_signal( int signum, siginfo_t * info, void * context )
{
    (get_instance()->handler_objects[info->si_fd])->receive_function( signum );
}

当然,您可以使用地图将文件描述符链接到对象。这绝不是全面的(或防弹的),而是在阅读你的问题时我想到的一个快速的想法。

祝你好运

阿尔文