summaryrefslogtreecommitdiffstats
path: root/cpukit/libcsupport/src/lseek.c
blob: 7c1f76b36bcf02669ea1e0954199e9c6b87d2182 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
/*
 *  lseek() - POSIX 1003.1b 6.5.3 - Reposition Read/Write File Offset
 *
 *  COPYRIGHT (c) 1989-1999.
 *  On-Line Applications Research Corporation (OAR).
 *
 *  The license and distribution terms for this file may be
 *  found in the file LICENSE in this distribution or at
 *  http://www.rtems.com/license/LICENSE.
 *
 *  $Id$
 */

#if HAVE_CONFIG_H
  #include "config.h"
#endif

#include <unistd.h>

#include <rtems/libio_.h>

off_t lseek( int fd, off_t offset, int whence )
{
  off_t rv = 0;
  rtems_libio_t *iop;
  off_t reference_offset;
  off_t old_offset;
  off_t new_offset;

  rtems_libio_check_fd( fd );
  iop = rtems_libio_iop( fd );
  rtems_libio_check_is_open(iop);

  old_offset = iop->offset;
  switch ( whence ) {
    case SEEK_SET:
      reference_offset = 0;
      break;
    case SEEK_CUR:
      reference_offset = old_offset;
      break;
    case SEEK_END:
      reference_offset = iop->size;
      break;
    default:
      errno = EINVAL;
      rv = (off_t) -1;
      break;
  }
  new_offset = reference_offset + offset;

  if ( rv == 0 ) {
    if (
      (reference_offset >= 0 && new_offset >= offset)
        || (reference_offset < 0 && new_offset <= offset)
    ) {
      switch ( rtems_filesystem_node_type( &iop->pathinfo ) ) {
        case RTEMS_FILESYSTEM_DIRECTORY:
        case RTEMS_FILESYSTEM_MEMORY_FILE:
          if ( new_offset < 0 ) {
            errno = EINVAL;
            rv = (off_t) -1;
          }
          break;
        default:
          break;
      }

      if ( rv == 0 ) {
        iop->offset = new_offset;
        rv = (*iop->pathinfo.handlers->lseek_h)( iop, offset, whence );
        if ( rv == (off_t) -1 ) {
          iop->offset = old_offset;
        }
      }
    } else {
      errno = EOVERFLOW;
      rv = (off_t) -1;
    }
  }

  return rv;
}

/*
 *  _lseek_r
 *
 *  This is the Newlib dependent reentrant version of lseek().
 */

#if defined(RTEMS_NEWLIB) && !defined(HAVE__LSEEK_R)

#include <reent.h>

off_t _lseek_r(
  struct _reent *ptr __attribute__((unused)),
  int            fd,
  off_t          offset,
  int            whence
)
{
  return lseek( fd, offset, whence );
}
#endif