OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [tags/] [gnu-src/] [gcc-4.5.1/] [gcc-4.5.1-or32-1.0rc4/] [libstdc++-v3/] [config/] [io/] [basic_file_stdio.cc] - Diff between revs 424 and 519

Only display areas with differences | Details | Blame | View Log

Rev 424 Rev 519
// Wrapper of C-language FILE struct -*- C++ -*-
// Wrapper of C-language FILE struct -*- C++ -*-
 
 
// Copyright (C) 2000, 2001, 2002, 2003, 2004, 2006, 2007, 2009, 2010
// Copyright (C) 2000, 2001, 2002, 2003, 2004, 2006, 2007, 2009, 2010
// Free Software Foundation, Inc.
// Free Software Foundation, Inc.
//
//
// This file is part of the GNU ISO C++ Library.  This library is free
// This file is part of the GNU ISO C++ Library.  This library is free
// software; you can redistribute it and/or modify it under the
// software; you can redistribute it and/or modify it under the
// terms of the GNU General Public License as published by the
// terms of the GNU General Public License as published by the
// Free Software Foundation; either version 3, or (at your option)
// Free Software Foundation; either version 3, or (at your option)
// any later version.
// any later version.
 
 
// This library is distributed in the hope that it will be useful,
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
// GNU General Public License for more details.
 
 
// Under Section 7 of GPL version 3, you are granted additional
// Under Section 7 of GPL version 3, you are granted additional
// permissions described in the GCC Runtime Library Exception, version
// permissions described in the GCC Runtime Library Exception, version
// 3.1, as published by the Free Software Foundation.
// 3.1, as published by the Free Software Foundation.
 
 
// You should have received a copy of the GNU General Public License and
// You should have received a copy of the GNU General Public License and
// a copy of the GCC Runtime Library Exception along with this program;
// a copy of the GCC Runtime Library Exception along with this program;
// see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
// see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
// <http://www.gnu.org/licenses/>.
// <http://www.gnu.org/licenses/>.
 
 
//
//
// ISO C++ 14882: 27.8  File-based streams
// ISO C++ 14882: 27.8  File-based streams
//
//
 
 
#include <bits/basic_file.h>
#include <bits/basic_file.h>
#include <fcntl.h>
#include <fcntl.h>
#include <errno.h>
#include <errno.h>
 
 
#ifdef _GLIBCXX_HAVE_POLL
#ifdef _GLIBCXX_HAVE_POLL
#include <poll.h>
#include <poll.h>
#endif
#endif
 
 
// Pick up ioctl on Solaris 2.8
// Pick up ioctl on Solaris 2.8
#ifdef _GLIBCXX_HAVE_UNISTD_H
#ifdef _GLIBCXX_HAVE_UNISTD_H
#include <unistd.h>
#include <unistd.h>
#endif
#endif
 
 
// Pick up FIONREAD on Solaris 2
// Pick up FIONREAD on Solaris 2
#ifdef _GLIBCXX_HAVE_SYS_IOCTL_H
#ifdef _GLIBCXX_HAVE_SYS_IOCTL_H
#define BSD_COMP 
#define BSD_COMP 
#include <sys/ioctl.h>
#include <sys/ioctl.h>
#endif
#endif
 
 
// Pick up FIONREAD on Solaris 2.5.
// Pick up FIONREAD on Solaris 2.5.
#ifdef _GLIBCXX_HAVE_SYS_FILIO_H
#ifdef _GLIBCXX_HAVE_SYS_FILIO_H
#include <sys/filio.h>
#include <sys/filio.h>
#endif
#endif
 
 
#ifdef _GLIBCXX_HAVE_SYS_UIO_H
#ifdef _GLIBCXX_HAVE_SYS_UIO_H
#include <sys/uio.h>
#include <sys/uio.h>
#endif
#endif
 
 
#if defined(_GLIBCXX_HAVE_S_ISREG) || defined(_GLIBCXX_HAVE_S_IFREG)
#if defined(_GLIBCXX_HAVE_S_ISREG) || defined(_GLIBCXX_HAVE_S_IFREG)
# include <sys/stat.h>
# include <sys/stat.h>
# ifdef _GLIBCXX_HAVE_S_ISREG
# ifdef _GLIBCXX_HAVE_S_ISREG
#  define _GLIBCXX_ISREG(x) S_ISREG(x)
#  define _GLIBCXX_ISREG(x) S_ISREG(x)
# else
# else
#  define _GLIBCXX_ISREG(x) (((x) & S_IFMT) == S_IFREG)
#  define _GLIBCXX_ISREG(x) (((x) & S_IFMT) == S_IFREG)
# endif
# endif
#endif
#endif
 
 
#include <limits> // For <off_t>::max() and min() and <streamsize>::max()
#include <limits> // For <off_t>::max() and min() and <streamsize>::max()
 
 
namespace
namespace
{
{
  // Map ios_base::openmode flags to a string for use in fopen().
  // Map ios_base::openmode flags to a string for use in fopen().
  // Table of valid combinations as given in [lib.filebuf.members]/2.
  // Table of valid combinations as given in [lib.filebuf.members]/2.
  static const char*
  static const char*
  fopen_mode(std::ios_base::openmode mode)
  fopen_mode(std::ios_base::openmode mode)
  {
  {
    enum
    enum
      {
      {
        in     = std::ios_base::in,
        in     = std::ios_base::in,
        out    = std::ios_base::out,
        out    = std::ios_base::out,
        trunc  = std::ios_base::trunc,
        trunc  = std::ios_base::trunc,
        app    = std::ios_base::app,
        app    = std::ios_base::app,
        binary = std::ios_base::binary
        binary = std::ios_base::binary
      };
      };
 
 
    // _GLIBCXX_RESOLVE_LIB_DEFECTS
    // _GLIBCXX_RESOLVE_LIB_DEFECTS
    // 596. 27.8.1.3 Table 112 omits "a+" and "a+b" modes.
    // 596. 27.8.1.3 Table 112 omits "a+" and "a+b" modes.
    switch (mode & (in|out|trunc|app|binary))
    switch (mode & (in|out|trunc|app|binary))
      {
      {
      case (   out                 ): return "w";
      case (   out                 ): return "w";
      case (   out      |app       ): return "a";
      case (   out      |app       ): return "a";
      case (             app       ): return "a";
      case (             app       ): return "a";
      case (   out|trunc           ): return "w";
      case (   out|trunc           ): return "w";
      case (in                     ): return "r";
      case (in                     ): return "r";
      case (in|out                 ): return "r+";
      case (in|out                 ): return "r+";
      case (in|out|trunc           ): return "w+";
      case (in|out|trunc           ): return "w+";
      case (in|out      |app       ): return "a+";
      case (in|out      |app       ): return "a+";
      case (in          |app       ): return "a+";
      case (in          |app       ): return "a+";
 
 
      case (   out          |binary): return "wb";
      case (   out          |binary): return "wb";
      case (   out      |app|binary): return "ab";
      case (   out      |app|binary): return "ab";
      case (             app|binary): return "ab";
      case (             app|binary): return "ab";
      case (   out|trunc    |binary): return "wb";
      case (   out|trunc    |binary): return "wb";
      case (in              |binary): return "rb";
      case (in              |binary): return "rb";
      case (in|out          |binary): return "r+b";
      case (in|out          |binary): return "r+b";
      case (in|out|trunc    |binary): return "w+b";
      case (in|out|trunc    |binary): return "w+b";
      case (in|out      |app|binary): return "a+b";
      case (in|out      |app|binary): return "a+b";
      case (in          |app|binary): return "a+b";
      case (in          |app|binary): return "a+b";
 
 
      default: return 0; // invalid
      default: return 0; // invalid
      }
      }
  }
  }
 
 
  // Wrapper handling partial write.
  // Wrapper handling partial write.
  static std::streamsize
  static std::streamsize
  xwrite(int __fd, const char* __s, std::streamsize __n)
  xwrite(int __fd, const char* __s, std::streamsize __n)
  {
  {
    std::streamsize __nleft = __n;
    std::streamsize __nleft = __n;
 
 
    for (;;)
    for (;;)
      {
      {
        const std::streamsize __ret = write(__fd, __s, __nleft);
        const std::streamsize __ret = write(__fd, __s, __nleft);
        if (__ret == -1L && errno == EINTR)
        if (__ret == -1L && errno == EINTR)
          continue;
          continue;
        if (__ret == -1L)
        if (__ret == -1L)
          break;
          break;
 
 
        __nleft -= __ret;
        __nleft -= __ret;
        if (__nleft == 0)
        if (__nleft == 0)
          break;
          break;
 
 
        __s += __ret;
        __s += __ret;
      }
      }
 
 
    return __n - __nleft;
    return __n - __nleft;
  }
  }
 
 
#ifdef _GLIBCXX_HAVE_WRITEV
#ifdef _GLIBCXX_HAVE_WRITEV
  // Wrapper handling partial writev.
  // Wrapper handling partial writev.
  static std::streamsize
  static std::streamsize
  xwritev(int __fd, const char* __s1, std::streamsize __n1,
  xwritev(int __fd, const char* __s1, std::streamsize __n1,
          const char* __s2, std::streamsize __n2)
          const char* __s2, std::streamsize __n2)
  {
  {
    std::streamsize __nleft = __n1 + __n2;
    std::streamsize __nleft = __n1 + __n2;
    std::streamsize __n1_left = __n1;
    std::streamsize __n1_left = __n1;
 
 
    struct iovec __iov[2];
    struct iovec __iov[2];
    __iov[1].iov_base = const_cast<char*>(__s2);
    __iov[1].iov_base = const_cast<char*>(__s2);
    __iov[1].iov_len = __n2;
    __iov[1].iov_len = __n2;
 
 
    for (;;)
    for (;;)
      {
      {
        __iov[0].iov_base = const_cast<char*>(__s1);
        __iov[0].iov_base = const_cast<char*>(__s1);
        __iov[0].iov_len = __n1_left;
        __iov[0].iov_len = __n1_left;
 
 
        const std::streamsize __ret = writev(__fd, __iov, 2);
        const std::streamsize __ret = writev(__fd, __iov, 2);
        if (__ret == -1L && errno == EINTR)
        if (__ret == -1L && errno == EINTR)
          continue;
          continue;
        if (__ret == -1L)
        if (__ret == -1L)
          break;
          break;
 
 
        __nleft -= __ret;
        __nleft -= __ret;
        if (__nleft == 0)
        if (__nleft == 0)
          break;
          break;
 
 
        const std::streamsize __off = __ret - __n1_left;
        const std::streamsize __off = __ret - __n1_left;
        if (__off >= 0)
        if (__off >= 0)
          {
          {
            __nleft -= xwrite(__fd, __s2 + __off, __n2 - __off);
            __nleft -= xwrite(__fd, __s2 + __off, __n2 - __off);
            break;
            break;
          }
          }
 
 
        __s1 += __ret;
        __s1 += __ret;
        __n1_left -= __ret;
        __n1_left -= __ret;
      }
      }
 
 
    return __n1 + __n2 - __nleft;
    return __n1 + __n2 - __nleft;
  }
  }
#endif
#endif
} // anonymous namespace
} // anonymous namespace
 
 
 
 
_GLIBCXX_BEGIN_NAMESPACE(std)
_GLIBCXX_BEGIN_NAMESPACE(std)
 
 
  // Definitions for __basic_file<char>.
  // Definitions for __basic_file<char>.
  __basic_file<char>::__basic_file(__c_lock* /*__lock*/) throw()
  __basic_file<char>::__basic_file(__c_lock* /*__lock*/) throw()
  : _M_cfile(NULL), _M_cfile_created(false) { }
  : _M_cfile(NULL), _M_cfile_created(false) { }
 
 
  __basic_file<char>::~__basic_file()
  __basic_file<char>::~__basic_file()
  { this->close(); }
  { this->close(); }
 
 
  __basic_file<char>*
  __basic_file<char>*
  __basic_file<char>::sys_open(__c_file* __file, ios_base::openmode)
  __basic_file<char>::sys_open(__c_file* __file, ios_base::openmode)
  {
  {
    __basic_file* __ret = NULL;
    __basic_file* __ret = NULL;
    if (!this->is_open() && __file)
    if (!this->is_open() && __file)
      {
      {
        int __err;
        int __err;
        errno = 0;
        errno = 0;
        do
        do
          __err = this->sync();
          __err = this->sync();
        while (__err && errno == EINTR);
        while (__err && errno == EINTR);
        if (!__err)
        if (!__err)
          {
          {
            _M_cfile = __file;
            _M_cfile = __file;
            _M_cfile_created = false;
            _M_cfile_created = false;
            __ret = this;
            __ret = this;
          }
          }
      }
      }
    return __ret;
    return __ret;
  }
  }
 
 
  __basic_file<char>*
  __basic_file<char>*
  __basic_file<char>::sys_open(int __fd, ios_base::openmode __mode) throw ()
  __basic_file<char>::sys_open(int __fd, ios_base::openmode __mode) throw ()
  {
  {
    __basic_file* __ret = NULL;
    __basic_file* __ret = NULL;
    const char* __c_mode = fopen_mode(__mode);
    const char* __c_mode = fopen_mode(__mode);
    if (__c_mode && !this->is_open() && (_M_cfile = fdopen(__fd, __c_mode)))
    if (__c_mode && !this->is_open() && (_M_cfile = fdopen(__fd, __c_mode)))
      {
      {
        char* __buf = NULL;
        char* __buf = NULL;
        _M_cfile_created = true;
        _M_cfile_created = true;
        if (__fd == 0)
        if (__fd == 0)
          setvbuf(_M_cfile, __buf, _IONBF, 0);
          setvbuf(_M_cfile, __buf, _IONBF, 0);
        __ret = this;
        __ret = this;
      }
      }
    return __ret;
    return __ret;
  }
  }
 
 
  __basic_file<char>*
  __basic_file<char>*
  __basic_file<char>::open(const char* __name, ios_base::openmode __mode,
  __basic_file<char>::open(const char* __name, ios_base::openmode __mode,
                           int /*__prot*/)
                           int /*__prot*/)
  {
  {
    __basic_file* __ret = NULL;
    __basic_file* __ret = NULL;
    const char* __c_mode = fopen_mode(__mode);
    const char* __c_mode = fopen_mode(__mode);
    if (__c_mode && !this->is_open())
    if (__c_mode && !this->is_open())
      {
      {
#ifdef _GLIBCXX_USE_LFS
#ifdef _GLIBCXX_USE_LFS
        if ((_M_cfile = fopen64(__name, __c_mode)))
        if ((_M_cfile = fopen64(__name, __c_mode)))
#else
#else
        if ((_M_cfile = fopen(__name, __c_mode)))
        if ((_M_cfile = fopen(__name, __c_mode)))
#endif
#endif
          {
          {
            _M_cfile_created = true;
            _M_cfile_created = true;
            __ret = this;
            __ret = this;
          }
          }
      }
      }
    return __ret;
    return __ret;
  }
  }
 
 
  bool
  bool
  __basic_file<char>::is_open() const throw ()
  __basic_file<char>::is_open() const throw ()
  { return _M_cfile != 0; }
  { return _M_cfile != 0; }
 
 
  int
  int
  __basic_file<char>::fd() throw ()
  __basic_file<char>::fd() throw ()
  { return fileno(_M_cfile); }
  { return fileno(_M_cfile); }
 
 
  __c_file*
  __c_file*
  __basic_file<char>::file() throw ()
  __basic_file<char>::file() throw ()
  { return _M_cfile; }
  { return _M_cfile; }
 
 
  __basic_file<char>*
  __basic_file<char>*
  __basic_file<char>::close()
  __basic_file<char>::close()
  {
  {
    __basic_file* __ret = static_cast<__basic_file*>(NULL);
    __basic_file* __ret = static_cast<__basic_file*>(NULL);
    if (this->is_open())
    if (this->is_open())
      {
      {
        int __err = 0;
        int __err = 0;
        if (_M_cfile_created)
        if (_M_cfile_created)
          {
          {
            // In general, no need to zero errno in advance if checking
            // In general, no need to zero errno in advance if checking
            // for error first. However, C89/C99 (at variance with IEEE
            // for error first. However, C89/C99 (at variance with IEEE
            // 1003.1, f.i.) do not mandate that fclose must set errno
            // 1003.1, f.i.) do not mandate that fclose must set errno
            // upon error.
            // upon error.
            errno = 0;
            errno = 0;
            do
            do
              __err = fclose(_M_cfile);
              __err = fclose(_M_cfile);
            while (__err && errno == EINTR);
            while (__err && errno == EINTR);
          }
          }
        _M_cfile = 0;
        _M_cfile = 0;
        if (!__err)
        if (!__err)
          __ret = this;
          __ret = this;
      }
      }
    return __ret;
    return __ret;
  }
  }
 
 
  streamsize
  streamsize
  __basic_file<char>::xsgetn(char* __s, streamsize __n)
  __basic_file<char>::xsgetn(char* __s, streamsize __n)
  {
  {
    streamsize __ret;
    streamsize __ret;
    do
    do
      __ret = read(this->fd(), __s, __n);
      __ret = read(this->fd(), __s, __n);
    while (__ret == -1L && errno == EINTR);
    while (__ret == -1L && errno == EINTR);
    return __ret;
    return __ret;
  }
  }
 
 
  streamsize
  streamsize
  __basic_file<char>::xsputn(const char* __s, streamsize __n)
  __basic_file<char>::xsputn(const char* __s, streamsize __n)
  { return xwrite(this->fd(), __s, __n); }
  { return xwrite(this->fd(), __s, __n); }
 
 
  streamsize
  streamsize
  __basic_file<char>::xsputn_2(const char* __s1, streamsize __n1,
  __basic_file<char>::xsputn_2(const char* __s1, streamsize __n1,
                               const char* __s2, streamsize __n2)
                               const char* __s2, streamsize __n2)
  {
  {
    streamsize __ret = 0;
    streamsize __ret = 0;
#ifdef _GLIBCXX_HAVE_WRITEV
#ifdef _GLIBCXX_HAVE_WRITEV
    __ret = xwritev(this->fd(), __s1, __n1, __s2, __n2);
    __ret = xwritev(this->fd(), __s1, __n1, __s2, __n2);
#else
#else
    if (__n1)
    if (__n1)
      __ret = xwrite(this->fd(), __s1, __n1);
      __ret = xwrite(this->fd(), __s1, __n1);
 
 
    if (__ret == __n1)
    if (__ret == __n1)
      __ret += xwrite(this->fd(), __s2, __n2);
      __ret += xwrite(this->fd(), __s2, __n2);
#endif
#endif
    return __ret;
    return __ret;
  }
  }
 
 
  streamoff
  streamoff
  __basic_file<char>::seekoff(streamoff __off, ios_base::seekdir __way) throw ()
  __basic_file<char>::seekoff(streamoff __off, ios_base::seekdir __way) throw ()
  {
  {
#ifdef _GLIBCXX_USE_LFS
#ifdef _GLIBCXX_USE_LFS
    return lseek64(this->fd(), __off, __way);
    return lseek64(this->fd(), __off, __way);
#else
#else
    if (__off > numeric_limits<off_t>::max()
    if (__off > numeric_limits<off_t>::max()
        || __off < numeric_limits<off_t>::min())
        || __off < numeric_limits<off_t>::min())
      return -1L;
      return -1L;
    return lseek(this->fd(), __off, __way);
    return lseek(this->fd(), __off, __way);
#endif
#endif
  }
  }
 
 
  int
  int
  __basic_file<char>::sync()
  __basic_file<char>::sync()
  { return fflush(_M_cfile); }
  { return fflush(_M_cfile); }
 
 
  streamsize
  streamsize
  __basic_file<char>::showmanyc()
  __basic_file<char>::showmanyc()
  {
  {
#ifdef FIONREAD
#ifdef FIONREAD
    // Pipes and sockets.    
    // Pipes and sockets.    
#ifdef _GLIBCXX_FIONREAD_TAKES_OFF_T
#ifdef _GLIBCXX_FIONREAD_TAKES_OFF_T
    off_t __num = 0;
    off_t __num = 0;
#else
#else
    int __num = 0;
    int __num = 0;
#endif
#endif
    int __r = ioctl(this->fd(), FIONREAD, &__num);
    int __r = ioctl(this->fd(), FIONREAD, &__num);
    if (!__r && __num >= 0)
    if (!__r && __num >= 0)
      return __num;
      return __num;
#endif    
#endif    
 
 
#ifdef _GLIBCXX_HAVE_POLL
#ifdef _GLIBCXX_HAVE_POLL
    // Cheap test.
    // Cheap test.
    struct pollfd __pfd[1];
    struct pollfd __pfd[1];
    __pfd[0].fd = this->fd();
    __pfd[0].fd = this->fd();
    __pfd[0].events = POLLIN;
    __pfd[0].events = POLLIN;
    if (poll(__pfd, 1, 0) <= 0)
    if (poll(__pfd, 1, 0) <= 0)
      return 0;
      return 0;
#endif   
#endif   
 
 
#if defined(_GLIBCXX_HAVE_S_ISREG) || defined(_GLIBCXX_HAVE_S_IFREG)
#if defined(_GLIBCXX_HAVE_S_ISREG) || defined(_GLIBCXX_HAVE_S_IFREG)
    // Regular files.
    // Regular files.
#ifdef _GLIBCXX_USE_LFS
#ifdef _GLIBCXX_USE_LFS
    struct stat64 __buffer;
    struct stat64 __buffer;
    const int __err = fstat64(this->fd(), &__buffer);
    const int __err = fstat64(this->fd(), &__buffer);
    if (!__err && _GLIBCXX_ISREG(__buffer.st_mode))
    if (!__err && _GLIBCXX_ISREG(__buffer.st_mode))
      {
      {
        const streamoff __off = __buffer.st_size - lseek64(this->fd(), 0,
        const streamoff __off = __buffer.st_size - lseek64(this->fd(), 0,
                                                           ios_base::cur);
                                                           ios_base::cur);
        return std::min(__off, streamoff(numeric_limits<streamsize>::max()));
        return std::min(__off, streamoff(numeric_limits<streamsize>::max()));
      }
      }
#else
#else
    struct stat __buffer;
    struct stat __buffer;
    const int __err = fstat(this->fd(), &__buffer);
    const int __err = fstat(this->fd(), &__buffer);
    if (!__err && _GLIBCXX_ISREG(__buffer.st_mode))
    if (!__err && _GLIBCXX_ISREG(__buffer.st_mode))
      return __buffer.st_size - lseek(this->fd(), 0, ios_base::cur);
      return __buffer.st_size - lseek(this->fd(), 0, ios_base::cur);
#endif
#endif
#endif
#endif
    return 0;
    return 0;
  }
  }
 
 
_GLIBCXX_END_NAMESPACE
_GLIBCXX_END_NAMESPACE
 
 
 
 

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.