summaryrefslogtreecommitdiff
path: root/libnetfs/file-lock.c
blob: 7bb81d43f5532e10ff6201aee11ae8699a16c268 (plain)
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
/*
   Copyright (C) 1995, 2015-2019 Free Software Foundation, Inc.
   Written by Michael I. Bushnell, p/BSG.

   This file is part of the GNU Hurd.

   The GNU Hurd 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.

   The GNU Hurd 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 the GNU Hurd.  If not, see <http://www.gnu.org/licenses/>.  */

#include "netfs.h"
#include "fs_S.h"

#include <fcntl.h>
#include <sys/file.h>

error_t
netfs_S_file_lock (struct protid *user,
		   int flags)
{
  error_t err;
  struct flock64 lock;
  struct node *node;
  int openstat = user->po->openstat;
  mach_port_t rendezvous = MACH_PORT_NULL;

  if (!user)
    return EOPNOTSUPP;

  lock.l_whence = SEEK_SET;
  lock.l_start = 0;
  lock.l_len = 0;

  if (flags & LOCK_UN)
    lock.l_type = F_UNLCK;
  else if (flags & LOCK_SH)
    lock.l_type = F_RDLCK;
  else if (flags & LOCK_EX)
    lock.l_type = F_WRLCK;
  else
    return EINVAL;

  /*
    XXX: Fix for flock(2) calling fcntl(2)
    From flock(2): A shared or exclusive lock can be placed on a file
    regardless of the mode in which the file was opened.
  */
  if (openstat & (O_RDONLY|O_WRONLY|O_EXEC)) openstat |= O_RDONLY|O_WRONLY;

  node = user->po->np;
  pthread_mutex_lock (&node->lock);
  err = fshelp_rlock_tweak (&node->userlock, &node->lock,
			    &user->po->lock_status, openstat,
			    0, 0, flags & LOCK_NB ? F_SETLK64 : F_SETLKW64,
			    &lock, rendezvous);
  pthread_mutex_unlock (&node->lock);

  return err;
}