⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 select.cxx

📁 eCos操作系统源码
💻 CXX
字号:
//==========================================================================////      select.cxx////      Fileio select() support////==========================================================================//####ECOSGPLCOPYRIGHTBEGIN####// -------------------------------------------// This file is part of eCos, the Embedded Configurable Operating System.// Copyright (C) 1998, 1999, 2000, 2001, 2002 Red Hat, Inc.// Copyright (C) 2002 Nick Garnett//// eCos 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 2 or (at your option) any later version.//// eCos 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 eCos; if not, write to the Free Software Foundation, Inc.,// 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.//// As a special exception, if other files instantiate templates or use macros// or inline functions from this file, or you compile this file and link it// with other works to produce a work based on this file, this file does not// by itself cause the resulting work to be covered by the GNU General Public// License. However the source code for this file must still be made available// in accordance with section (3) of the GNU General Public License.//// This exception does not invalidate any other reasons why a work based on// this file might be covered by the GNU General Public License.//// Alternative licenses for eCos may be arranged by contacting Red Hat, Inc.// at http://sources.redhat.com/ecos/ecos-license/// -------------------------------------------//####ECOSGPLCOPYRIGHTEND####//==========================================================================//#####DESCRIPTIONBEGIN####//// Author(s):           nickg// Contributors:        nickg// Date:                2000-05-25// Purpose:             Fileio select() support// Description:         Support for select().//                      //              //              ////####DESCRIPTIONEND####////==========================================================================#include <pkgconf/hal.h>#include <pkgconf/kernel.h>#include <pkgconf/io_fileio.h>#include <cyg/kernel/ktypes.h>         // base kernel types#include <cyg/infra/cyg_trac.h>        // tracing macros#include <cyg/infra/cyg_ass.h>         // assertion macros#include <stdarg.h>                    // for fcntl()#include "fio.h"                       // Private header#include <sys/select.h>                // select header#include <cyg/kernel/sched.hxx>        // scheduler definitions#include <cyg/kernel/thread.hxx>       // thread definitions#include <cyg/kernel/mutex.hxx>        // mutex definitions#include <cyg/kernel/clock.hxx>        // clock definitions#include <cyg/kernel/sched.inl>#include <cyg/kernel/thread.inl>#include <cyg/kernel/clock.inl>//==========================================================================// File object locking#define LOCK_FILE( fp ) cyg_file_lock( fp )#define UNLOCK_FILE( fp ) cyg_file_unlock( fp )//==========================================================================// Local variables// Mutex for serializing select processing. This essntially controls// access to the contents of the selinfo structures embedded in the// client system data structures.static Cyg_Mutex select_mutex CYGBLD_ATTRIB_INIT_PRI(CYG_INIT_IO_FS);// Condition variable where any thread that is waiting for a select to// fire is suspended. Note that select is not intended to be a real time// operation. Whenever any selectable event occurs, all selecting threads// will be resumed. They must then rescan their selectees and resuspend if// necessary.static Cyg_Condition_Variable selwait( select_mutex ) CYGBLD_ATTRIB_INIT_PRI(CYG_INIT_IO_FS);static volatile cyg_uint32 selwake_count = 0;//==========================================================================// Timeval to ticks conversion support// Converters from sec and us to ticksstatic struct Cyg_Clock::converter us_converter, sec_converter;static cyg_bool converters_initialized = false;externC cyg_tick_count cyg_timeval_to_ticks( const struct timeval *tv ){    if( !converters_initialized )    {        // Create the converters we need.        Cyg_Clock::real_time_clock->get_other_to_clock_converter( 1000, &us_converter );        Cyg_Clock::real_time_clock->get_other_to_clock_converter( 1000000000, &sec_converter );        converters_initialized = true;    }        // Short circuit zero timeval    if( tv->tv_sec == 0 && tv->tv_usec == 0 )    {        return 0;    }            // Convert the seconds field to ticks.    cyg_tick_count ticks = Cyg_Clock::convert( tv->tv_sec, &sec_converter );    // Convert the nanoseconds. This will round down to nearest whole tick.    ticks += Cyg_Clock::convert( (cyg_tick_count)tv->tv_usec, &us_converter );    return ticks;}//==========================================================================// Select API functionstatic intcyg_pselect(int nfd, fd_set *in, fd_set *out, fd_set *ex,           struct timeval *tv, const sigset_t *mask){    FILEIO_ENTRY();    int error = ENOERR;    int fd, mode, num;    cyg_file *fp;    fd_set in_res, out_res, ex_res;  // Result sets    fd_set *selection[3], *result[3];    cyg_tick_count ticks;    int mode_type[] = {CYG_FREAD, CYG_FWRITE, 0};    cyg_uint32 wake_count;    sigset_t oldmask;    FD_ZERO(&in_res);    FD_ZERO(&out_res);    FD_ZERO(&ex_res);    // Set up sets    selection[0] = in;   result[0] = &in_res;    selection[1] = out;  result[1] = &out_res;    selection[2] = ex;   result[2] = &ex_res;    // Compute end time    if (tv)        ticks = cyg_timeval_to_ticks( tv );    else ticks = 0;    // Lock the mutex    select_mutex.lock();    // Scan sets for possible I/O until something found, timeout or error.    while (!error)    {        wake_count = selwake_count;                num = 0;  // Total file descriptors "ready"        for (mode = 0;  !error && mode < 3;  mode++)        {            if (selection[mode]) {                for (fd = 0;  !error && fd < nfd;  fd++)                {                    if (FD_ISSET(fd, selection[mode]))                    {                        fp = cyg_fp_get( fd );                        if( fp == NULL )                        {                            error = EBADF;                            break;                        }                        if ((*fp->f_ops->fo_select)(fp, mode_type[mode], 0))                        {                            FD_SET(fd, result[mode]);                            num++;                        }                        cyg_fp_free( fp );                    }                }            }        }                if (num)        {            // Found something, update user's sets            if (in)  FD_COPY( &in_res, in );            if (out) FD_COPY( &out_res, out );            if (ex)  FD_COPY( &ex_res, ex );            select_mutex.unlock();            CYG_FILEIO_DELIVER_SIGNALS( mask );            FILEIO_RETURN_VALUE(num);        }        Cyg_Scheduler::lock();        // Switch to the supplied signal mask. This will permit delivery        // of any signals that might terminate this select operation.                CYG_FILEIO_SIGMASK_SET( mask, &oldmask );            do        {            // We need to see if any signals have been posted while we            // were testing all those files. The handlers will not            // have run because we have ASRs inhibited but the signal            // will have been set pending.            if( CYG_FILEIO_SIGPENDING() )            {                // There are pending signals so we need to terminate                // the select operation and return EINTR. Handlers for                // the pending signals will be called just before we                // return.                error = EINTR;                break;            }                        if( wake_count == selwake_count )            {                // Nothing found, see if we want to wait                if (tv)                {                    // Special case of "poll"                    if (ticks == 0)                    {                        error = EAGAIN;                        break;                    }                    ticks += Cyg_Clock::real_time_clock->current_value();                                    if( !selwait.wait( ticks ) )                    {                        // A non-standard wakeup, if the current time is equal to                        // or past the timeout, return zero. Otherwise return                        // EINTR, since we have been released.                        if( Cyg_Clock::real_time_clock->current_value() >= ticks )                        {                            error = EAGAIN;                            break;                        }                        else error = EINTR;                    }                    ticks -= Cyg_Clock::real_time_clock->current_value();                }                else                {                    // Wait forever (until something happens)                                if( !selwait.wait() )                        error = EINTR;                }            }        } while(0);        CYG_FILEIO_SIGMASK_SET( &oldmask, NULL );                Cyg_Scheduler::unlock();            } // while(!error)    select_mutex.unlock();     // If the error code is EAGAIN, this means that a timeout has    // happened. We return zero in that case, rather than a proper    // error code.    // If the error code is EINTR, then a signal may be pending    // delivery. Call back into the POSIX package to handle it.        if( error == EAGAIN )        FILEIO_RETURN_VALUE(0);    else if( error == EINTR )        CYG_FILEIO_DELIVER_SIGNALS( mask );    FILEIO_RETURN(error);}// -------------------------------------------------------------------------// Select API function__externC intselect(int nfd, fd_set *in, fd_set *out, fd_set *ex, struct timeval *tv){	return cyg_pselect(nfd, in, out, ex, tv, NULL);}// -------------------------------------------------------------------------// Pselect API function//// This is derived from the POSIX-200X specification.__externC intpselect(int nfd, fd_set *in, fd_set *out, fd_set *ex,	const struct timespec *ts, const sigset_t *sigmask){	struct timeval tv;#ifndef CYGPKG_POSIX_SIGNALS        CYG_ASSERT( sigmask == NULL,                    "pselect called with non-null sigmask without POSIX signal support"                    );#endif	if (ts != NULL)        {            tv.tv_sec = ts->tv_sec;            tv.tv_usec = ts->tv_nsec/1000;        }	return cyg_pselect(nfd, in, out, ex, ts ? &tv : NULL, sigmask);}//==========================================================================// Select support functions.// -------------------------------------------------------------------------// cyg_selinit() is used to initialize a selinfo structurevoid cyg_selinit( struct CYG_SELINFO_TAG *sip ){    sip->si_info = 0;    sip->si_thread = 0;}// -------------------------------------------------------------------------// cyg_selrecord() is called when a client device needs to register// the current thread for selection.void cyg_selrecord( CYG_ADDRWORD info, struct CYG_SELINFO_TAG *sip ){    sip->si_info = info;    sip->si_thread = (CYG_ADDRESS)Cyg_Thread::self();}// -------------------------------------------------------------------------// cyg_selwakeup() is called when the client device matches the select// criterion, and needs to wake up a selector.void cyg_selwakeup( struct CYG_SELINFO_TAG *sip ){    // We don't actually use the si_info field of selinfo at present.    // A potential use would be to select one of several selwait condition    // variables to signal. However, that would only be necessary if we    // end up having lots of threads in select.    Cyg_Scheduler::lock();     if( sip->si_thread != 0 )    {        // If the thread pointer is still present, this selection has        // not been fired before. We just wake up all threads waiting,        // regardless of whether they are waiting for this event or        // not.  This avoids any race conditions, and is consistent        // with the behaviour of the BSD kernel.                sip->si_thread = 0;        selwait.broadcast();        selwake_count++;    }    Cyg_Scheduler::unlock();    }// -------------------------------------------------------------------------// EOF select.cxx

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -