2013-09-10 11:35:01 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
2000-01-13 12:09:36 +00:00
byte range locking code
2000-04-25 20:30:58 +00:00
Updated to handle range splits / merges .
Copyright ( C ) Andrew Tridgell 1992 - 2000
Copyright ( C ) Jeremy Allison 1992 - 2000
2011-05-28 10:24:20 +02:00
2000-01-13 12:09:36 +00:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
2000-01-13 12:09:36 +00:00
( at your option ) any later version .
2011-05-28 10:24:20 +02:00
2000-01-13 12:09:36 +00:00
This program 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 .
2011-05-28 10:24:20 +02:00
2000-01-13 12:09:36 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2000-01-13 12:09:36 +00:00
*/
2000-04-25 20:30:58 +00:00
/* This module implements a tdb based byte range locking service,
2000-01-13 12:09:36 +00:00
replacing the fcntl ( ) based byte range locking previously
used . This allows us to provide the same semantics as NT */
# include "includes.h"
2011-02-25 23:20:06 +01:00
# include "system/filesys.h"
2011-03-23 12:43:17 +01:00
# include "locking/proto.h"
2010-07-04 20:20:44 +02:00
# include "smbd/globals.h"
2011-07-07 17:42:08 +02:00
# include "dbwrap/dbwrap.h"
2011-07-06 16:40:21 +02:00
# include "dbwrap/dbwrap_open.h"
2011-02-24 23:05:57 +01:00
# include "serverid.h"
2011-03-24 15:31:06 +01:00
# include "messages.h"
2013-03-13 14:47:18 +01:00
# include "util_tdb.h"
2000-01-13 12:09:36 +00:00
2005-04-27 18:32:37 +00:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_LOCKING
2001-08-27 08:19:43 +00:00
# define ZERO_ZERO 0
2000-04-25 20:30:58 +00:00
/* The open brlock.tdb database. */
2000-01-13 12:09:36 +00:00
2007-05-27 10:35:14 +00:00
static struct db_context * brlock_db ;
2000-01-13 12:09:36 +00:00
2013-09-10 19:42:06 +02:00
struct byte_range_lock {
struct files_struct * fsp ;
unsigned int num_locks ;
bool modified ;
2013-09-11 12:48:14 +00:00
bool have_read_oplocks ;
2013-09-10 19:42:06 +02:00
struct lock_struct * lock_data ;
struct db_record * record ;
} ;
2006-04-10 15:33:04 +00:00
/****************************************************************************
Debug info at level 10 for lock struct .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2012-06-28 09:54:14 +02:00
static void print_lock_struct ( unsigned int i , const struct lock_struct * pls )
2006-04-10 15:33:04 +00:00
{
2010-05-07 06:20:50 -07:00
DEBUG ( 10 , ( " [%u]: smblctx = %llu, tid = %u, pid = %s, " ,
2006-04-10 15:33:04 +00:00
i ,
2010-05-07 06:20:50 -07:00
( unsigned long long ) pls - > context . smblctx ,
2006-04-10 15:33:04 +00:00
( unsigned int ) pls - > context . tid ,
2011-06-08 14:05:55 +10:00
server_id_str ( talloc_tos ( ) , & pls - > context . pid ) ) ) ;
2011-05-28 10:24:20 +02:00
2014-07-11 12:52:06 +02:00
DEBUG ( 10 , ( " start = %ju, size = %ju, fnum = %ju, %s %s \n " ,
( uintmax_t ) pls - > start ,
( uintmax_t ) pls - > size ,
( uintmax_t ) pls - > fnum ,
lock_type_name ( pls - > lock_type ) ,
lock_flav_name ( pls - > lock_flav ) ) ) ;
2006-04-10 15:33:04 +00:00
}
2013-09-10 19:40:43 +02:00
unsigned int brl_num_locks ( const struct byte_range_lock * brl )
{
return brl - > num_locks ;
}
2013-09-10 19:41:32 +02:00
struct files_struct * brl_fsp ( struct byte_range_lock * brl )
{
return brl - > fsp ;
}
2013-09-11 12:48:14 +00:00
bool brl_have_read_oplocks ( const struct byte_range_lock * brl )
{
return brl - > have_read_oplocks ;
}
void brl_set_have_read_oplocks ( struct byte_range_lock * brl ,
bool have_read_oplocks )
{
2013-09-13 14:13:51 +02:00
DEBUG ( 10 , ( " Setting have_read_oplocks to %s \n " ,
have_read_oplocks ? " true " : " false " ) ) ;
2013-09-11 12:48:14 +00:00
SMB_ASSERT ( brl - > record ! = NULL ) ; /* otherwise we're readonly */
brl - > have_read_oplocks = have_read_oplocks ;
brl - > modified = true ;
}
2000-01-13 12:09:36 +00:00
/****************************************************************************
2000-04-25 20:30:58 +00:00
See if two locking contexts are equal .
2000-01-13 12:09:36 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-25 20:30:58 +00:00
2013-09-10 11:39:52 +00:00
static bool brl_same_context ( const struct lock_context * ctx1 ,
2006-04-10 15:33:04 +00:00
const struct lock_context * ctx2 )
2000-01-13 12:09:36 +00:00
{
2012-06-16 00:26:26 +02:00
return ( serverid_equal ( & ctx1 - > pid , & ctx2 - > pid ) & &
2010-05-07 06:20:50 -07:00
( ctx1 - > smblctx = = ctx2 - > smblctx ) & &
2005-09-30 17:13:37 +00:00
( ctx1 - > tid = = ctx2 - > tid ) ) ;
2000-01-13 12:09:36 +00:00
}
2004-10-18 22:01:10 +00:00
/****************************************************************************
See if lck1 and lck2 overlap .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
static bool brl_overlap ( const struct lock_struct * lck1 ,
2006-04-10 15:33:04 +00:00
const struct lock_struct * lck2 )
2004-10-18 22:01:10 +00:00
{
2009-02-20 15:22:15 -08:00
/* XXX Remove for Win7 compatibility. */
2013-02-18 10:02:51 +01:00
/* this extra check is not redundant - it copes with locks
2004-10-18 22:01:10 +00:00
that go beyond the end of 64 bit file space */
if ( lck1 - > size ! = 0 & &
lck1 - > start = = lck2 - > start & &
lck1 - > size = = lck2 - > size ) {
return True ;
}
if ( lck1 - > start > = ( lck2 - > start + lck2 - > size ) | |
lck2 - > start > = ( lck1 - > start + lck1 - > size ) ) {
return False ;
}
return True ;
}
2000-01-13 12:09:36 +00:00
/****************************************************************************
2000-04-25 20:30:58 +00:00
See if lock2 can be added when lock1 is in place .
2000-01-13 12:09:36 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-25 20:30:58 +00:00
2013-09-10 11:35:01 +00:00
static bool brl_conflict ( const struct lock_struct * lck1 ,
2006-04-10 15:33:04 +00:00
const struct lock_struct * lck2 )
2001-08-24 21:09:38 +00:00
{
2006-04-10 15:33:04 +00:00
/* Ignore PENDING locks. */
2006-07-29 19:14:24 +00:00
if ( IS_PENDING_LOCK ( lck1 - > lock_type ) | | IS_PENDING_LOCK ( lck2 - > lock_type ) )
2003-02-27 01:04:34 +00:00
return False ;
2006-04-10 15:33:04 +00:00
/* Read locks never conflict. */
2001-08-24 21:09:38 +00:00
if ( lck1 - > lock_type = = READ_LOCK & & lck2 - > lock_type = = READ_LOCK ) {
return False ;
}
2009-02-20 15:22:15 -08:00
/* A READ lock can stack on top of a WRITE lock if they have the same
* context & fnum . */
if ( lck1 - > lock_type = = WRITE_LOCK & & lck2 - > lock_type = = READ_LOCK & &
brl_same_context ( & lck1 - > context , & lck2 - > context ) & &
lck1 - > fnum = = lck2 - > fnum ) {
2001-08-24 21:09:38 +00:00
return False ;
}
2004-10-18 22:01:10 +00:00
return brl_overlap ( lck1 , lck2 ) ;
2013-09-10 11:35:01 +00:00
}
2001-08-24 21:09:38 +00:00
2006-04-10 15:33:04 +00:00
/****************************************************************************
See if lock2 can be added when lock1 is in place - when both locks are POSIX
flavour . POSIX locks ignore fnum - they only care about dev / ino which we
know already match .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2013-09-10 11:35:01 +00:00
static bool brl_conflict_posix ( const struct lock_struct * lck1 ,
2006-04-10 15:33:04 +00:00
const struct lock_struct * lck2 )
{
# if defined(DEVELOPER)
SMB_ASSERT ( lck1 - > lock_flav = = POSIX_LOCK ) ;
SMB_ASSERT ( lck2 - > lock_flav = = POSIX_LOCK ) ;
# endif
/* Ignore PENDING locks. */
2006-07-29 19:14:24 +00:00
if ( IS_PENDING_LOCK ( lck1 - > lock_type ) | | IS_PENDING_LOCK ( lck2 - > lock_type ) )
2006-04-10 15:33:04 +00:00
return False ;
/* Read locks never conflict. */
if ( lck1 - > lock_type = = READ_LOCK & & lck2 - > lock_type = = READ_LOCK ) {
return False ;
}
2014-01-29 00:55:19 +01:00
/* Locks on the same context don't conflict. Ignore fnum. */
2006-04-10 15:33:04 +00:00
if ( brl_same_context ( & lck1 - > context , & lck2 - > context ) ) {
return False ;
}
/* One is read, the other write, or the context is different,
do they overlap ? */
return brl_overlap ( lck1 , lck2 ) ;
2013-09-10 11:35:01 +00:00
}
2006-04-10 15:33:04 +00:00
2001-08-27 08:19:43 +00:00
# if ZERO_ZERO
2013-09-10 11:35:01 +00:00
static bool brl_conflict1 ( const struct lock_struct * lck1 ,
2006-04-10 15:33:04 +00:00
const struct lock_struct * lck2 )
2001-08-27 08:19:43 +00:00
{
2006-07-29 19:14:24 +00:00
if ( IS_PENDING_LOCK ( lck1 - > lock_type ) | | IS_PENDING_LOCK ( lck2 - > lock_type ) )
2003-02-27 01:04:34 +00:00
return False ;
2001-08-27 08:19:43 +00:00
if ( lck1 - > lock_type = = READ_LOCK & & lck2 - > lock_type = = READ_LOCK ) {
return False ;
}
if ( brl_same_context ( & lck1 - > context , & lck2 - > context ) & &
lck2 - > lock_type = = READ_LOCK & & lck1 - > fnum = = lck2 - > fnum ) {
return False ;
}
if ( lck2 - > start = = 0 & & lck2 - > size = = 0 & & lck1 - > size ! = 0 ) {
return True ;
}
if ( lck1 - > start > = ( lck2 - > start + lck2 - > size ) | |
lck2 - > start > = ( lck1 - > start + lck1 - > size ) ) {
return False ;
}
2011-05-28 10:24:20 +02:00
2001-08-27 08:19:43 +00:00
return True ;
2013-09-10 11:35:01 +00:00
}
2001-08-27 08:19:43 +00:00
# endif
2001-08-24 21:09:38 +00:00
/****************************************************************************
Check to see if this lock conflicts , but ignore our own locks on the
2006-04-10 15:33:04 +00:00
same fnum only . This is the read / write lock check code path .
This is never used in the POSIX lock case .
2001-08-24 21:09:38 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2014-07-11 13:21:20 +02:00
static bool brl_conflict_other ( const struct lock_struct * lock ,
const struct lock_struct * rw_probe )
2000-01-13 12:09:36 +00:00
{
2014-07-11 13:21:20 +02:00
if ( IS_PENDING_LOCK ( lock - > lock_type ) | |
IS_PENDING_LOCK ( rw_probe - > lock_type ) ) {
2003-02-27 01:04:34 +00:00
return False ;
2014-07-11 13:21:20 +02:00
}
2003-02-27 01:04:34 +00:00
2014-07-11 13:21:20 +02:00
if ( lock - > lock_type = = READ_LOCK & & rw_probe - > lock_type = = READ_LOCK ) {
2000-01-14 08:01:44 +00:00
return False ;
2014-07-11 13:21:20 +02:00
}
2000-01-13 12:09:36 +00:00
2014-07-11 13:21:20 +02:00
if ( lock - > lock_flav = = POSIX_LOCK & &
rw_probe - > lock_flav = = POSIX_LOCK ) {
2014-07-11 13:27:39 +02:00
/*
* POSIX flavour locks never conflict here - this is only called
* in the read / write path .
*/
2006-04-10 15:33:04 +00:00
return False ;
2014-07-11 13:21:20 +02:00
}
2006-04-10 15:33:04 +00:00
2014-07-11 13:27:39 +02:00
if ( ! brl_overlap ( lock , rw_probe ) ) {
/*
* I / O can only conflict when overlapping a lock , thus let it
* pass
*/
return false ;
}
if ( ! brl_same_context ( & lock - > context , & rw_probe - > context ) ) {
/*
* Different process , conflict
*/
return true ;
}
2003-02-22 01:09:57 +00:00
2014-07-11 13:27:39 +02:00
if ( lock - > fnum ! = rw_probe - > fnum ) {
/*
* Different file handle , conflict
*/
return true ;
2003-02-22 01:09:57 +00:00
}
2000-01-13 12:09:36 +00:00
2014-07-11 13:27:39 +02:00
if ( ( lock - > lock_type = = READ_LOCK ) & &
( rw_probe - > lock_type = = WRITE_LOCK ) ) {
/*
* Incoming WRITE locks conflict with existing READ locks even
* if the context is the same . JRA . See LOCKTEST7 in
* smbtorture .
*/
return true ;
}
/*
* I / O request compatible with existing lock , let it pass without
* conflict
*/
return false ;
2013-09-10 11:35:01 +00:00
}
2000-01-13 12:09:36 +00:00
2006-07-29 19:14:24 +00:00
/****************************************************************************
Check if an unlock overlaps a pending lock .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
static bool brl_pending_overlap ( const struct lock_struct * lock , const struct lock_struct * pend_lock )
2006-07-29 19:14:24 +00:00
{
if ( ( lock - > start < = pend_lock - > start ) & & ( lock - > start + lock - > size > pend_lock - > start ) )
return True ;
2014-07-01 13:30:50 -07:00
if ( ( lock - > start > = pend_lock - > start ) & & ( lock - > start < pend_lock - > start + pend_lock - > size ) )
2006-07-29 19:14:24 +00:00
return True ;
return False ;
}
2004-10-18 22:01:10 +00:00
/****************************************************************************
2006-07-17 21:09:02 +00:00
Amazingly enough , w2k3 " remembers " whether the last lock failure on a fnum
2004-10-18 22:01:10 +00:00
is the same as this one and changes its error code . I wonder if any
app depends on this ?
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2013-09-10 11:41:39 +00:00
static NTSTATUS brl_lock_failed ( files_struct * fsp ,
const struct lock_struct * lock ,
bool blocking_lock )
2004-10-18 22:01:10 +00:00
{
2006-07-17 21:09:02 +00:00
if ( lock - > start > = 0xEF000000 & & ( lock - > start > > 63 ) = = 0 ) {
2004-10-18 22:01:10 +00:00
/* amazing the little things you learn with a test
suite . Locks beyond this offset ( as a 64 bit
number ! ) always generate the conflict error code ,
unless the top bit is set */
2006-07-18 01:05:51 +00:00
if ( ! blocking_lock ) {
2006-07-17 21:09:02 +00:00
fsp - > last_lock_failure = * lock ;
}
return NT_STATUS_FILE_LOCK_CONFLICT ;
}
2012-06-16 00:26:26 +02:00
if ( serverid_equal ( & lock - > context . pid , & fsp - > last_lock_failure . context . pid ) & &
2006-07-17 21:09:02 +00:00
lock - > context . tid = = fsp - > last_lock_failure . context . tid & &
lock - > fnum = = fsp - > last_lock_failure . fnum & &
lock - > start = = fsp - > last_lock_failure . start ) {
2004-10-18 22:01:10 +00:00
return NT_STATUS_FILE_LOCK_CONFLICT ;
}
2006-07-17 21:09:02 +00:00
2006-07-18 01:05:51 +00:00
if ( ! blocking_lock ) {
2006-07-17 21:09:02 +00:00
fsp - > last_lock_failure = * lock ;
}
2004-10-18 22:01:10 +00:00
return NT_STATUS_LOCK_NOT_GRANTED ;
}
2000-01-13 12:09:36 +00:00
/****************************************************************************
2006-04-10 15:33:04 +00:00
Open up the brlock . tdb database .
2000-01-13 12:09:36 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-05-22 20:35:48 +00:00
2007-12-05 20:53:22 +01:00
void brl_init ( bool read_only )
2000-05-03 14:29:05 +00:00
{
2009-11-16 09:40:47 +01:00
int tdb_flags ;
2007-05-27 10:35:14 +00:00
if ( brlock_db ) {
2006-04-10 15:33:04 +00:00
return ;
}
2009-11-16 09:40:47 +01:00
2010-09-27 05:46:07 -07:00
tdb_flags = TDB_DEFAULT | TDB_VOLATILE | TDB_CLEAR_IF_FIRST | TDB_INCOMPATIBLE_HASH ;
2009-11-16 09:40:47 +01:00
if ( ! lp_clustering ( ) ) {
/*
* We can ' t use the SEQNUM trick to cache brlock
* entries in the clustering case because ctdb seqnum
* propagation has a delay .
*/
tdb_flags | = TDB_SEQNUM ;
}
2008-01-16 12:09:48 +03:00
brlock_db = db_open ( NULL , lock_path ( " brlock.tdb " ) ,
2014-01-10 15:41:08 +13:00
SMB_OPEN_DATABASE_TDB_HASH_SIZE , tdb_flags ,
2012-01-06 17:19:54 +01:00
read_only ? O_RDONLY : ( O_RDWR | O_CREAT ) , 0644 ,
2014-01-27 14:49:12 +01:00
DBWRAP_LOCK_ORDER_2 , DBWRAP_FLAG_NONE ) ;
2007-05-27 10:35:14 +00:00
if ( ! brlock_db ) {
2006-04-10 15:33:04 +00:00
DEBUG ( 0 , ( " Failed to open byte range locking database %s \n " ,
lock_path ( " brlock.tdb " ) ) ) ;
return ;
}
}
2000-05-03 14:29:05 +00:00
2006-04-10 15:33:04 +00:00
/****************************************************************************
Close down the brlock . tdb database .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-05-03 14:29:05 +00:00
2007-12-05 20:53:22 +01:00
void brl_shutdown ( void )
2006-04-10 15:33:04 +00:00
{
2007-05-27 10:35:14 +00:00
TALLOC_FREE ( brlock_db ) ;
2006-04-10 15:33:04 +00:00
}
2000-05-03 14:29:05 +00:00
2006-04-10 15:33:04 +00:00
# if ZERO_ZERO
/****************************************************************************
Compare two locks for sorting .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-05-22 20:35:48 +00:00
2013-09-10 11:35:01 +00:00
static int lock_compare ( const struct lock_struct * lck1 ,
2006-04-10 15:33:04 +00:00
const struct lock_struct * lck2 )
{
if ( lck1 - > start ! = lck2 - > start ) {
return ( lck1 - > start - lck2 - > start ) ;
}
if ( lck2 - > size ! = lck1 - > size ) {
return ( ( int ) lck1 - > size - ( int ) lck2 - > size ) ;
}
return 0 ;
}
# endif
2001-05-22 20:35:48 +00:00
2006-04-10 15:33:04 +00:00
/****************************************************************************
Lock a range of bytes - Windows lock semantics .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-05-22 20:35:48 +00:00
2009-02-09 21:51:29 -08:00
NTSTATUS brl_lock_windows_default ( struct byte_range_lock * br_lck ,
struct lock_struct * plock , bool blocking_lock )
2006-04-10 15:33:04 +00:00
{
unsigned int i ;
files_struct * fsp = br_lck - > fsp ;
2007-05-05 20:43:06 +00:00
struct lock_struct * locks = br_lck - > lock_data ;
2009-02-03 11:56:35 -08:00
NTSTATUS status ;
2006-04-10 15:33:04 +00:00
2009-02-09 21:51:29 -08:00
SMB_ASSERT ( plock - > lock_type ! = UNLOCK_LOCK ) ;
2010-05-05 15:57:57 -07:00
if ( ( plock - > start + plock - > size - 1 < plock - > start ) & &
plock - > size ! = 0 ) {
return NT_STATUS_INVALID_LOCK_RANGE ;
}
2009-12-04 14:04:08 -08:00
2010-05-05 15:57:57 -07:00
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
2006-04-10 15:33:04 +00:00
/* Do any Windows or POSIX locks conflict ? */
if ( brl_conflict ( & locks [ i ] , plock ) ) {
2007-05-19 20:57:12 +00:00
/* Remember who blocked us. */
2010-05-07 06:20:50 -07:00
plock - > context . smblctx = locks [ i ] . context . smblctx ;
2006-07-18 01:05:51 +00:00
return brl_lock_failed ( fsp , plock , blocking_lock ) ;
2006-04-10 15:33:04 +00:00
}
# if ZERO_ZERO
2013-09-10 11:35:01 +00:00
if ( plock - > start = = 0 & & plock - > size = = 0 & &
2006-04-10 15:33:04 +00:00
locks [ i ] . size = = 0 ) {
break ;
2001-05-22 20:35:48 +00:00
}
2006-04-10 15:33:04 +00:00
# endif
}
2001-05-22 20:35:48 +00:00
2009-02-03 11:56:35 -08:00
if ( ! IS_PENDING_LOCK ( plock - > lock_type ) ) {
contend_level2_oplocks_begin ( fsp , LEVEL2_CONTEND_WINDOWS_BRL ) ;
}
2006-04-10 15:33:04 +00:00
/* We can get the Windows lock, now see if it needs to
be mapped into a lower level POSIX one , and if so can
2006-07-11 18:01:26 +00:00
we get it ? */
2000-05-03 14:29:05 +00:00
2006-11-11 17:05:11 +00:00
if ( ! IS_PENDING_LOCK ( plock - > lock_type ) & & lp_posix_locking ( fsp - > conn - > params ) ) {
2006-07-11 18:01:26 +00:00
int errno_ret ;
if ( ! set_posix_lock_windows_flavour ( fsp ,
plock - > start ,
plock - > size ,
plock - > lock_type ,
& plock - > context ,
locks ,
br_lck - > num_locks ,
& errno_ret ) ) {
2007-05-19 20:57:12 +00:00
/* We don't know who blocked us. */
2010-05-07 06:20:50 -07:00
plock - > context . smblctx = 0xFFFFFFFFFFFFFFFFLL ;
2007-05-19 20:57:12 +00:00
2006-07-11 18:01:26 +00:00
if ( errno_ret = = EACCES | | errno_ret = = EAGAIN ) {
2009-02-03 11:56:35 -08:00
status = NT_STATUS_FILE_LOCK_CONFLICT ;
goto fail ;
2006-04-10 15:33:04 +00:00
} else {
2009-02-03 11:56:35 -08:00
status = map_nt_error_from_unix ( errno ) ;
goto fail ;
2006-04-10 15:33:04 +00:00
}
2000-05-03 14:29:05 +00:00
}
}
2000-04-25 20:30:58 +00:00
2006-04-10 15:33:04 +00:00
/* no conflicts - add it to the list of locks */
2013-09-10 21:04:47 +02:00
locks = talloc_realloc ( br_lck , locks , struct lock_struct ,
( br_lck - > num_locks + 1 ) ) ;
2006-04-10 15:33:04 +00:00
if ( ! locks ) {
2009-02-03 11:56:35 -08:00
status = NT_STATUS_NO_MEMORY ;
goto fail ;
2000-05-03 14:29:05 +00:00
}
2006-04-10 15:33:04 +00:00
memcpy ( & locks [ br_lck - > num_locks ] , plock , sizeof ( struct lock_struct ) ) ;
br_lck - > num_locks + = 1 ;
2007-05-05 20:43:06 +00:00
br_lck - > lock_data = locks ;
2006-04-10 15:33:04 +00:00
br_lck - > modified = True ;
return NT_STATUS_OK ;
2009-02-03 11:56:35 -08:00
fail :
if ( ! IS_PENDING_LOCK ( plock - > lock_type ) ) {
contend_level2_oplocks_end ( fsp , LEVEL2_CONTEND_WINDOWS_BRL ) ;
}
return status ;
2000-05-03 14:29:05 +00:00
}
/****************************************************************************
2006-04-10 15:33:04 +00:00
Cope with POSIX range splits and merges .
2000-05-03 14:29:05 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-05-22 20:35:48 +00:00
2009-10-05 10:27:48 -07:00
static unsigned int brlock_posix_split_merge ( struct lock_struct * lck_arr , /* Output array. */
struct lock_struct * ex , /* existing lock. */
struct lock_struct * plock ) /* proposed lock. */
2000-01-13 12:09:36 +00:00
{
2007-10-18 17:40:25 -07:00
bool lock_types_differ = ( ex - > lock_type ! = plock - > lock_type ) ;
2006-04-10 15:33:04 +00:00
/* We can't merge non-conflicting locks on different context - ignore fnum. */
if ( ! brl_same_context ( & ex - > context , & plock - > context ) ) {
/* Just copy. */
memcpy ( & lck_arr [ 0 ] , ex , sizeof ( struct lock_struct ) ) ;
return 1 ;
2000-05-03 14:29:05 +00:00
}
2006-04-10 15:33:04 +00:00
/* We now know we have the same context. */
/* Did we overlap ? */
/*********************************************
2009-10-05 10:27:48 -07:00
+ - - - - - - - - - +
| ex |
+ - - - - - - - - - +
+ - - - - - - - +
| plock |
+ - - - - - - - +
2006-04-10 15:33:04 +00:00
OR . . . .
2009-10-05 10:27:48 -07:00
+ - - - - - - - - - +
| ex |
+ - - - - - - - - - +
2006-04-10 15:33:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-07-11 18:01:26 +00:00
if ( ( ex - > start > ( plock - > start + plock - > size ) ) | |
2009-10-05 10:27:48 -07:00
( plock - > start > ( ex - > start + ex - > size ) ) ) {
2006-04-10 15:33:04 +00:00
/* No overlap with this lock - copy existing. */
2009-10-05 10:27:48 -07:00
2006-04-10 15:33:04 +00:00
memcpy ( & lck_arr [ 0 ] , ex , sizeof ( struct lock_struct ) ) ;
return 1 ;
2002-07-15 10:35:28 +00:00
}
2006-04-10 15:33:04 +00:00
/*********************************************
2006-07-11 18:01:26 +00:00
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
| ex |
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
2006-04-10 15:33:04 +00:00
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
| plock | - > replace with plock .
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
2009-10-05 10:27:48 -07:00
OR
+ - - - - - - - - - - - - - - - +
| ex |
+ - - - - - - - - - - - - - - - +
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
| plock | - > replace with plock .
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
2006-04-10 15:33:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
if ( ( ex - > start > = plock - > start ) & &
2009-10-05 10:27:48 -07:00
( ex - > start + ex - > size < = plock - > start + plock - > size ) ) {
/* Replace - discard existing lock. */
return 0 ;
2006-04-10 15:33:04 +00:00
}
/*********************************************
2009-10-05 10:27:48 -07:00
Adjacent after .
+ - - - - - - - +
| ex |
+ - - - - - - - +
+ - - - - - - - - - - - - - - - +
| plock |
+ - - - - - - - - - - - - - - - +
BECOMES . . . .
+ - - - - - - - - - - - - - - - + - - - - - - - +
| plock | ex | - different lock types .
+ - - - - - - - - - - - - - - - + - - - - - - - +
OR . . . . ( merge )
+ - - - - - - - - - - - - - - - - - - - - - - - +
| plock | - same lock type .
+ - - - - - - - - - - - - - - - - - - - - - - - +
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
if ( plock - > start + plock - > size = = ex - > start ) {
/* If the lock types are the same, we merge, if different, we
add the remainder of the old lock . */
if ( lock_types_differ ) {
/* Add existing. */
memcpy ( & lck_arr [ 0 ] , ex , sizeof ( struct lock_struct ) ) ;
return 1 ;
} else {
/* Merge - adjust incoming lock as we may have more
* merging to come . */
plock - > size + = ex - > size ;
return 0 ;
}
}
/*********************************************
Adjacent before .
+ - - - - - - - +
| ex |
+ - - - - - - - +
+ - - - - - - - - - - - - - - - +
| plock |
+ - - - - - - - - - - - - - - - +
BECOMES . . . .
+ - - - - - - - + - - - - - - - - - - - - - - - +
| ex | plock | - different lock types
+ - - - - - - - + - - - - - - - - - - - - - - - +
OR . . . . ( merge )
+ - - - - - - - - - - - - - - - - - - - - - - - +
| plock | - same lock type .
+ - - - - - - - - - - - - - - - - - - - - - - - +
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
if ( ex - > start + ex - > size = = plock - > start ) {
/* If the lock types are the same, we merge, if different, we
add the existing lock . */
if ( lock_types_differ ) {
memcpy ( & lck_arr [ 0 ] , ex , sizeof ( struct lock_struct ) ) ;
return 1 ;
} else {
/* Merge - adjust incoming lock as we may have more
* merging to come . */
plock - > start = ex - > start ;
plock - > size + = ex - > size ;
return 0 ;
}
}
/*********************************************
Overlap after .
2006-07-11 18:01:26 +00:00
+ - - - - - - - - - - - - - - - - - - - - - - - +
| ex |
+ - - - - - - - - - - - - - - - - - - - - - - - +
2006-04-10 15:33:04 +00:00
+ - - - - - - - - - - - - - - - +
| plock |
+ - - - - - - - - - - - - - - - +
2009-10-05 10:27:48 -07:00
OR
+ - - - - - - - - - - - - - - - - +
| ex |
+ - - - - - - - - - - - - - - - - +
2006-07-11 18:01:26 +00:00
+ - - - - - - - - - - - - - - - +
| plock |
+ - - - - - - - - - - - - - - - +
2006-04-10 15:33:04 +00:00
BECOMES . . . .
+ - - - - - - - - - - - - - - - + - - - - - - - +
| plock | ex | - different lock types .
+ - - - - - - - - - - - - - - - + - - - - - - - +
2006-07-11 18:01:26 +00:00
OR . . . . ( merge )
2006-04-10 15:33:04 +00:00
+ - - - - - - - - - - - - - - - - - - - - - - - +
2009-10-05 10:27:48 -07:00
| plock | - same lock type .
2006-04-10 15:33:04 +00:00
+ - - - - - - - - - - - - - - - - - - - - - - - +
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
if ( ( ex - > start > = plock - > start ) & &
2009-10-05 10:27:48 -07:00
( ex - > start < = plock - > start + plock - > size ) & &
( ex - > start + ex - > size > plock - > start + plock - > size ) ) {
2006-04-10 15:33:04 +00:00
/* If the lock types are the same, we merge, if different, we
2009-10-05 10:27:48 -07:00
add the remainder of the old lock . */
2006-04-10 15:33:04 +00:00
if ( lock_types_differ ) {
2009-10-05 10:27:48 -07:00
/* Add remaining existing. */
memcpy ( & lck_arr [ 0 ] , ex , sizeof ( struct lock_struct ) ) ;
2006-04-10 15:33:04 +00:00
/* Adjust existing start and size. */
2009-10-05 10:27:48 -07:00
lck_arr [ 0 ] . start = plock - > start + plock - > size ;
lck_arr [ 0 ] . size = ( ex - > start + ex - > size ) - ( plock - > start + plock - > size ) ;
2006-04-10 15:33:04 +00:00
return 1 ;
2009-10-05 10:27:48 -07:00
} else {
/* Merge - adjust incoming lock as we may have more
* merging to come . */
plock - > size + = ( ex - > start + ex - > size ) - ( plock - > start + plock - > size ) ;
return 0 ;
2006-04-10 15:33:04 +00:00
}
}
/*********************************************
2009-10-05 10:27:48 -07:00
Overlap before .
+ - - - - - - - - - - - - - - - - - - - - - - - +
| ex |
+ - - - - - - - - - - - - - - - - - - - - - - - +
+ - - - - - - - - - - - - - - - +
| plock |
+ - - - - - - - - - - - - - - - +
OR
+ - - - - - - - - - - - - - +
| ex |
+ - - - - - - - - - - - - - +
+ - - - - - - - - - - - - - - - +
| plock |
+ - - - - - - - - - - - - - - - +
2006-04-10 15:33:04 +00:00
BECOMES . . . .
2009-10-05 10:27:48 -07:00
+ - - - - - - - + - - - - - - - - - - - - - - - +
| ex | plock | - different lock types
+ - - - - - - - + - - - - - - - - - - - - - - - +
2006-04-10 15:33:04 +00:00
2006-07-11 18:01:26 +00:00
OR . . . . ( merge )
2009-10-05 10:27:48 -07:00
+ - - - - - - - - - - - - - - - - - - - - - - - +
| plock | - same lock type .
+ - - - - - - - - - - - - - - - - - - - - - - - +
2006-04-10 15:33:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
if ( ( ex - > start < plock - > start ) & &
2006-07-11 18:01:26 +00:00
( ex - > start + ex - > size > = plock - > start ) & &
2006-04-10 15:33:04 +00:00
( ex - > start + ex - > size < = plock - > start + plock - > size ) ) {
/* If the lock types are the same, we merge, if different, we
2009-10-05 10:27:48 -07:00
add the truncated old lock . */
2006-04-10 15:33:04 +00:00
if ( lock_types_differ ) {
memcpy ( & lck_arr [ 0 ] , ex , sizeof ( struct lock_struct ) ) ;
/* Adjust existing size. */
lck_arr [ 0 ] . size = plock - > start - ex - > start ;
return 1 ;
2009-10-05 10:27:48 -07:00
} else {
/* Merge - adjust incoming lock as we may have more
* merging to come . MUST ADJUST plock SIZE FIRST ! */
plock - > size + = ( plock - > start - ex - > start ) ;
plock - > start = ex - > start ;
return 0 ;
2006-04-10 15:33:04 +00:00
}
}
/*********************************************
2009-10-05 10:27:48 -07:00
Complete overlap .
2006-04-10 15:33:04 +00:00
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
| ex |
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
+ - - - - - - - - - +
| plock |
+ - - - - - - - - - +
BECOMES . . . . .
+ - - - - - - - + - - - - - - - - - + - - - - - - - - - +
| ex | plock | ex | - different lock types .
+ - - - - - - - + - - - - - - - - - + - - - - - - - - - +
OR
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
2009-10-05 10:27:48 -07:00
| plock | - same lock type .
2006-04-10 15:33:04 +00:00
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - +
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
if ( ( ex - > start < plock - > start ) & & ( ex - > start + ex - > size > plock - > start + plock - > size ) ) {
if ( lock_types_differ ) {
/* We have to split ex into two locks here. */
memcpy ( & lck_arr [ 0 ] , ex , sizeof ( struct lock_struct ) ) ;
2009-10-05 10:27:48 -07:00
memcpy ( & lck_arr [ 1 ] , ex , sizeof ( struct lock_struct ) ) ;
2006-04-10 15:33:04 +00:00
/* Adjust first existing size. */
lck_arr [ 0 ] . size = plock - > start - ex - > start ;
/* Adjust second existing start and size. */
2009-10-05 10:27:48 -07:00
lck_arr [ 1 ] . start = plock - > start + plock - > size ;
lck_arr [ 1 ] . size = ( ex - > start + ex - > size ) - ( plock - > start + plock - > size ) ;
return 2 ;
2006-04-10 15:33:04 +00:00
} else {
2009-10-05 10:27:48 -07:00
/* Just eat the existing locks, merge them into plock. */
plock - > start = ex - > start ;
plock - > size = ex - > size ;
return 0 ;
2006-04-10 15:33:04 +00:00
}
}
/* Never get here. */
2007-06-15 21:58:49 +00:00
smb_panic ( " brlock_posix_split_merge " ) ;
2006-04-10 15:33:04 +00:00
/* Notreached. */
2007-06-15 21:58:49 +00:00
2006-04-25 21:36:35 +00:00
/* Keep some compilers happy. */
return 0 ;
2000-01-13 12:09:36 +00:00
}
2001-05-22 20:35:48 +00:00
/****************************************************************************
2006-04-10 15:33:04 +00:00
Lock a range of bytes - POSIX lock semantics .
We must cope with range splits and merges .
2001-05-22 20:35:48 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-05-14 13:01:28 +00:00
static NTSTATUS brl_lock_posix ( struct messaging_context * msg_ctx ,
struct byte_range_lock * br_lck ,
2007-05-19 20:57:12 +00:00
struct lock_struct * plock )
2001-05-22 20:35:48 +00:00
{
2009-02-03 11:56:35 -08:00
unsigned int i , count , posix_count ;
2007-05-05 20:43:06 +00:00
struct lock_struct * locks = br_lck - > lock_data ;
2006-04-10 15:33:04 +00:00
struct lock_struct * tp ;
2007-10-18 17:40:25 -07:00
bool signal_pending_read = False ;
2009-02-03 11:56:35 -08:00
bool break_oplocks = false ;
NTSTATUS status ;
2006-04-10 15:33:04 +00:00
/* No zero-zero locks for POSIX. */
if ( plock - > start = = 0 & & plock - > size = = 0 ) {
return NT_STATUS_INVALID_PARAMETER ;
}
/* Don't allow 64-bit lock wrap. */
2010-05-05 15:57:57 -07:00
if ( plock - > start + plock - > size - 1 < plock - > start ) {
2006-04-10 15:33:04 +00:00
return NT_STATUS_INVALID_PARAMETER ;
}
/* The worst case scenario here is we have to split an
existing POSIX lock range into two , and add our lock ,
so we need at most 2 more entries . */
2013-09-10 21:04:47 +02:00
tp = talloc_array ( br_lck , struct lock_struct , br_lck - > num_locks + 2 ) ;
2006-04-10 15:33:04 +00:00
if ( ! tp ) {
return NT_STATUS_NO_MEMORY ;
}
2009-10-05 10:27:48 -07:00
2009-02-03 11:56:35 -08:00
count = posix_count = 0 ;
2009-10-05 10:27:48 -07:00
2006-04-10 15:33:04 +00:00
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
2006-07-29 19:14:24 +00:00
struct lock_struct * curr_lock = & locks [ i ] ;
/* If we have a pending read lock, a lock downgrade should
trigger a lock re - evaluation . */
if ( curr_lock - > lock_type = = PENDING_READ_LOCK & &
brl_pending_overlap ( plock , curr_lock ) ) {
signal_pending_read = True ;
}
if ( curr_lock - > lock_flav = = WINDOWS_LOCK ) {
2006-04-10 15:33:04 +00:00
/* Do any Windows flavour locks conflict ? */
2006-07-29 19:14:24 +00:00
if ( brl_conflict ( curr_lock , plock ) ) {
2006-04-10 15:33:04 +00:00
/* No games with error messages. */
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( tp ) ;
2007-05-19 20:57:12 +00:00
/* Remember who blocked us. */
2010-05-07 06:20:50 -07:00
plock - > context . smblctx = curr_lock - > context . smblctx ;
2006-04-10 15:33:04 +00:00
return NT_STATUS_FILE_LOCK_CONFLICT ;
}
/* Just copy the Windows lock into the new array. */
2006-07-29 19:14:24 +00:00
memcpy ( & tp [ count ] , curr_lock , sizeof ( struct lock_struct ) ) ;
2006-04-10 15:33:04 +00:00
count + + ;
} else {
2009-02-10 18:09:12 +01:00
unsigned int tmp_count = 0 ;
2009-02-03 11:56:35 -08:00
2006-04-10 15:33:04 +00:00
/* POSIX conflict semantics are different. */
2006-07-29 19:14:24 +00:00
if ( brl_conflict_posix ( curr_lock , plock ) ) {
2006-04-10 15:33:04 +00:00
/* Can't block ourselves with POSIX locks. */
/* No games with error messages. */
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( tp ) ;
2007-05-19 20:57:12 +00:00
/* Remember who blocked us. */
2010-05-07 06:20:50 -07:00
plock - > context . smblctx = curr_lock - > context . smblctx ;
2006-04-10 15:33:04 +00:00
return NT_STATUS_FILE_LOCK_CONFLICT ;
}
/* Work out overlaps. */
2009-10-05 10:27:48 -07:00
tmp_count + = brlock_posix_split_merge ( & tp [ count ] , curr_lock , plock ) ;
2009-02-03 11:56:35 -08:00
posix_count + = tmp_count ;
count + = tmp_count ;
2006-04-10 15:33:04 +00:00
}
}
2009-02-03 11:56:35 -08:00
/*
* Break oplocks while we hold a brl . Since lock ( ) and unlock ( ) calls
* are not symetric with POSIX semantics , we cannot guarantee our
* contend_level2_oplocks_begin / end calls will be acquired and
* released one - for - one as with Windows semantics . Therefore we only
* call contend_level2_oplocks_begin if this is the first POSIX brl on
* the file .
*/
break_oplocks = ( ! IS_PENDING_LOCK ( plock - > lock_type ) & &
posix_count = = 0 ) ;
if ( break_oplocks ) {
contend_level2_oplocks_begin ( br_lck - > fsp ,
LEVEL2_CONTEND_POSIX_BRL ) ;
}
2009-10-05 10:27:48 -07:00
/* Try and add the lock in order, sorted by lock start. */
for ( i = 0 ; i < count ; i + + ) {
struct lock_struct * curr_lock = & tp [ i ] ;
if ( curr_lock - > start < = plock - > start ) {
continue ;
}
}
if ( i < count ) {
memmove ( & tp [ i + 1 ] , & tp [ i ] ,
( count - i ) * sizeof ( struct lock_struct ) ) ;
2006-07-11 18:01:26 +00:00
}
2009-10-05 10:27:48 -07:00
memcpy ( & tp [ i ] , plock , sizeof ( struct lock_struct ) ) ;
count + + ;
2006-07-11 18:01:26 +00:00
2006-04-10 15:33:04 +00:00
/* We can get the POSIX lock, now see if it needs to
be mapped into a lower level POSIX one , and if so can
2006-07-11 18:01:26 +00:00
we get it ? */
2006-04-10 15:33:04 +00:00
2006-11-11 17:05:11 +00:00
if ( ! IS_PENDING_LOCK ( plock - > lock_type ) & & lp_posix_locking ( br_lck - > fsp - > conn - > params ) ) {
2006-07-11 18:01:26 +00:00
int errno_ret ;
2001-05-22 20:35:48 +00:00
2006-07-11 18:01:26 +00:00
/* The lower layer just needs to attempt to
get the system POSIX lock . We ' ve weeded out
any conflicts above . */
2006-04-10 15:33:04 +00:00
2006-07-11 18:01:26 +00:00
if ( ! set_posix_lock_posix_flavour ( br_lck - > fsp ,
plock - > start ,
plock - > size ,
plock - > lock_type ,
& errno_ret ) ) {
2007-05-19 20:57:12 +00:00
/* We don't know who blocked us. */
2010-05-07 06:20:50 -07:00
plock - > context . smblctx = 0xFFFFFFFFFFFFFFFFLL ;
2007-05-19 20:57:12 +00:00
2006-07-11 18:01:26 +00:00
if ( errno_ret = = EACCES | | errno_ret = = EAGAIN ) {
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( tp ) ;
2009-02-03 11:56:35 -08:00
status = NT_STATUS_FILE_LOCK_CONFLICT ;
goto fail ;
2006-04-10 15:33:04 +00:00
} else {
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( tp ) ;
2009-02-03 11:56:35 -08:00
status = map_nt_error_from_unix ( errno ) ;
goto fail ;
2006-04-10 15:33:04 +00:00
}
}
2002-07-15 10:35:28 +00:00
}
2000-01-13 12:09:36 +00:00
2009-10-05 10:27:48 -07:00
/* If we didn't use all the allocated size,
* Realloc so we don ' t leak entries per lock call . */
if ( count < br_lck - > num_locks + 2 ) {
2013-09-10 21:04:47 +02:00
tp = talloc_realloc ( br_lck , tp , struct lock_struct , count ) ;
2009-10-05 10:27:48 -07:00
if ( ! tp ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
2001-08-27 08:19:43 +00:00
}
2009-10-05 10:27:48 -07:00
2006-04-10 15:33:04 +00:00
br_lck - > num_locks = count ;
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( br_lck - > lock_data ) ;
2007-05-05 20:43:06 +00:00
br_lck - > lock_data = tp ;
2006-07-29 19:14:24 +00:00
locks = tp ;
2006-04-10 15:33:04 +00:00
br_lck - > modified = True ;
2006-07-29 19:14:24 +00:00
/* A successful downgrade from write to read lock can trigger a lock
re - evalutation where waiting readers can now proceed . */
if ( signal_pending_read ) {
/* Send unlock messages to any pending read waiters that overlap. */
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
struct lock_struct * pend_lock = & locks [ i ] ;
/* Ignore non-pending locks. */
if ( ! IS_PENDING_LOCK ( pend_lock - > lock_type ) ) {
continue ;
}
if ( pend_lock - > lock_type = = PENDING_READ_LOCK & &
brl_pending_overlap ( plock , pend_lock ) ) {
DEBUG ( 10 , ( " brl_lock_posix: sending unlock message to pid %s \n " ,
procid_str_static ( & pend_lock - > context . pid ) ) ) ;
2007-05-14 13:01:28 +00:00
messaging_send ( msg_ctx , pend_lock - > context . pid ,
MSG_SMB_UNLOCK , & data_blob_null ) ;
2006-07-29 19:14:24 +00:00
}
}
}
2006-04-10 15:33:04 +00:00
return NT_STATUS_OK ;
2009-02-03 11:56:35 -08:00
fail :
if ( break_oplocks ) {
contend_level2_oplocks_end ( br_lck - > fsp ,
LEVEL2_CONTEND_POSIX_BRL ) ;
}
return status ;
2001-08-27 08:19:43 +00:00
}
2009-07-23 20:28:58 -04:00
NTSTATUS smb_vfs_call_brl_lock_windows ( struct vfs_handle_struct * handle ,
struct byte_range_lock * br_lck ,
struct lock_struct * plock ,
2014-07-03 13:51:45 +00:00
bool blocking_lock )
2009-07-23 20:28:58 -04:00
{
VFS_FIND ( brl_lock_windows ) ;
2011-12-03 20:45:04 -08:00
return handle - > fns - > brl_lock_windows_fn ( handle , br_lck , plock ,
2014-07-03 13:51:45 +00:00
blocking_lock ) ;
2009-07-23 20:28:58 -04:00
}
2000-01-13 12:09:36 +00:00
/****************************************************************************
2000-04-25 20:30:58 +00:00
Lock a range of bytes .
2000-01-13 12:09:36 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-25 20:30:58 +00:00
2007-05-14 13:01:28 +00:00
NTSTATUS brl_lock ( struct messaging_context * msg_ctx ,
struct byte_range_lock * br_lck ,
2010-05-07 06:20:50 -07:00
uint64_t smblctx ,
2007-05-07 09:35:35 +00:00
struct server_id pid ,
2006-04-10 15:33:04 +00:00
br_off start ,
2013-09-10 11:35:01 +00:00
br_off size ,
2006-04-10 15:33:04 +00:00
enum brl_type lock_type ,
enum brl_flavour lock_flav ,
2007-10-18 17:40:25 -07:00
bool blocking_lock ,
2014-07-03 14:05:03 +00:00
uint64_t * psmblctx )
2000-01-13 12:09:36 +00:00
{
2006-04-10 15:33:04 +00:00
NTSTATUS ret ;
struct lock_struct lock ;
2000-01-13 12:09:36 +00:00
2001-08-27 08:19:43 +00:00
# if !ZERO_ZERO
2001-08-24 21:09:38 +00:00
if ( start = = 0 & & size = = 0 ) {
2001-08-27 08:19:43 +00:00
DEBUG ( 0 , ( " client sent 0/0 lock - please report this \n " ) ) ;
2001-08-24 21:09:38 +00:00
}
# endif
2014-07-11 12:41:54 +02:00
lock = ( struct lock_struct ) {
. context . smblctx = smblctx ,
. context . pid = pid ,
. context . tid = br_lck - > fsp - > conn - > cnum ,
. start = start ,
. size = size ,
. fnum = br_lck - > fsp - > fnum ,
. lock_type = lock_type ,
. lock_flav = lock_flav
} ;
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
if ( lock_flav = = WINDOWS_LOCK ) {
2009-02-09 21:51:29 -08:00
ret = SMB_VFS_BRL_LOCK_WINDOWS ( br_lck - > fsp - > conn , br_lck ,
2014-07-03 13:51:45 +00:00
& lock , blocking_lock ) ;
2006-04-10 15:33:04 +00:00
} else {
2007-05-14 13:01:28 +00:00
ret = brl_lock_posix ( msg_ctx , br_lck , & lock ) ;
2000-01-13 12:09:36 +00:00
}
2001-08-27 08:19:43 +00:00
# if ZERO_ZERO
/* sort the lock list */
2010-02-14 10:02:35 +11:00
TYPESAFE_QSORT ( br_lck - > lock_data , ( size_t ) br_lck - > num_locks , lock_compare ) ;
2001-08-27 08:19:43 +00:00
# endif
2007-05-19 20:57:12 +00:00
/* If we're returning an error, return who blocked us. */
2010-05-07 06:20:50 -07:00
if ( ! NT_STATUS_IS_OK ( ret ) & & psmblctx ) {
* psmblctx = lock . context . smblctx ;
2007-05-19 20:57:12 +00:00
}
2006-04-10 15:33:04 +00:00
return ret ;
2000-01-13 12:09:36 +00:00
}
2014-06-29 11:31:18 +02:00
static void brl_delete_lock_struct ( struct lock_struct * locks ,
unsigned num_locks ,
unsigned del_idx )
{
if ( del_idx > = num_locks ) {
return ;
}
memmove ( & locks [ del_idx ] , & locks [ del_idx + 1 ] ,
sizeof ( * locks ) * ( num_locks - del_idx - 1 ) ) ;
}
2000-01-13 12:09:36 +00:00
/****************************************************************************
2006-04-10 15:33:04 +00:00
Unlock a range of bytes - Windows semantics .
2000-01-13 12:09:36 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-25 20:30:58 +00:00
2009-02-09 21:51:29 -08:00
bool brl_unlock_windows_default ( struct messaging_context * msg_ctx ,
2007-05-14 13:01:28 +00:00
struct byte_range_lock * br_lck ,
const struct lock_struct * plock )
2000-01-13 12:09:36 +00:00
{
2006-04-10 15:33:04 +00:00
unsigned int i , j ;
2007-05-05 20:43:06 +00:00
struct lock_struct * locks = br_lck - > lock_data ;
2006-07-11 18:01:26 +00:00
enum brl_type deleted_lock_type = READ_LOCK ; /* shut the compiler up.... */
2000-01-13 12:09:36 +00:00
2009-02-09 21:51:29 -08:00
SMB_ASSERT ( plock - > lock_type = = UNLOCK_LOCK ) ;
2006-04-10 15:33:04 +00:00
# if ZERO_ZERO
2006-07-11 18:01:26 +00:00
/* Delete write locks by preference... The lock list
is sorted in the zero zero case . */
2006-04-10 15:33:04 +00:00
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
2006-07-15 00:34:08 +00:00
struct lock_struct * lock = & locks [ i ] ;
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
if ( lock - > lock_type = = WRITE_LOCK & &
brl_same_context ( & lock - > context , & plock - > context ) & &
lock - > fnum = = plock - > fnum & &
lock - > lock_flav = = WINDOWS_LOCK & &
lock - > start = = plock - > start & &
lock - > size = = plock - > size ) {
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
/* found it - delete it */
2006-07-11 18:01:26 +00:00
deleted_lock_type = lock - > lock_type ;
break ;
2006-04-10 15:33:04 +00:00
}
2000-04-27 22:23:04 +00:00
}
2006-07-11 18:01:26 +00:00
if ( i ! = br_lck - > num_locks ) {
/* We found it - don't search again. */
goto unlock_continue ;
}
2006-04-10 15:33:04 +00:00
# endif
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
2006-07-15 00:34:08 +00:00
struct lock_struct * lock = & locks [ i ] ;
2000-01-13 12:09:36 +00:00
2010-05-06 09:07:49 -07:00
if ( IS_PENDING_LOCK ( lock - > lock_type ) ) {
continue ;
}
2006-04-10 15:33:04 +00:00
/* Only remove our own locks that match in start, size, and flavour. */
if ( brl_same_context ( & lock - > context , & plock - > context ) & &
lock - > fnum = = plock - > fnum & &
lock - > lock_flav = = WINDOWS_LOCK & &
lock - > start = = plock - > start & &
lock - > size = = plock - > size ) {
2006-07-11 18:01:26 +00:00
deleted_lock_type = lock - > lock_type ;
2006-04-10 15:33:04 +00:00
break ;
}
}
2000-04-25 20:30:58 +00:00
2006-04-10 15:33:04 +00:00
if ( i = = br_lck - > num_locks ) {
/* we didn't find it */
return False ;
}
2000-04-25 20:30:58 +00:00
2006-07-11 18:01:26 +00:00
# if ZERO_ZERO
unlock_continue :
# endif
2014-06-29 11:31:18 +02:00
brl_delete_lock_struct ( locks , br_lck - > num_locks , i ) ;
2006-07-11 18:01:26 +00:00
br_lck - > num_locks - = 1 ;
br_lck - > modified = True ;
/* Unlock the underlying POSIX regions. */
2006-11-11 17:05:11 +00:00
if ( lp_posix_locking ( br_lck - > fsp - > conn - > params ) ) {
2006-07-11 18:01:26 +00:00
release_posix_lock_windows_flavour ( br_lck - > fsp ,
plock - > start ,
plock - > size ,
deleted_lock_type ,
& plock - > context ,
locks ,
br_lck - > num_locks ) ;
2006-04-10 15:33:04 +00:00
}
2003-04-04 20:38:12 +00:00
2006-04-10 15:33:04 +00:00
/* Send unlock messages to any pending waiters that overlap. */
for ( j = 0 ; j < br_lck - > num_locks ; j + + ) {
struct lock_struct * pend_lock = & locks [ j ] ;
2003-04-04 20:38:12 +00:00
2006-04-10 15:33:04 +00:00
/* Ignore non-pending locks. */
2006-07-29 19:14:24 +00:00
if ( ! IS_PENDING_LOCK ( pend_lock - > lock_type ) ) {
2006-04-10 15:33:04 +00:00
continue ;
}
2000-04-27 22:23:04 +00:00
2006-04-10 15:33:04 +00:00
/* We could send specific lock info here... */
2006-07-15 00:34:08 +00:00
if ( brl_pending_overlap ( plock , pend_lock ) ) {
2006-04-10 15:33:04 +00:00
DEBUG ( 10 , ( " brl_unlock: sending unlock message to pid %s \n " ,
procid_str_static ( & pend_lock - > context . pid ) ) ) ;
2007-05-14 13:01:28 +00:00
messaging_send ( msg_ctx , pend_lock - > context . pid ,
MSG_SMB_UNLOCK , & data_blob_null ) ;
2000-04-27 22:23:04 +00:00
}
2001-08-24 21:09:38 +00:00
}
2009-02-03 11:56:35 -08:00
contend_level2_oplocks_end ( br_lck - > fsp , LEVEL2_CONTEND_WINDOWS_BRL ) ;
2006-04-10 15:33:04 +00:00
return True ;
}
/****************************************************************************
Unlock a range of bytes - POSIX semantics .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-02-27 01:04:34 +00:00
2007-10-18 17:40:25 -07:00
static bool brl_unlock_posix ( struct messaging_context * msg_ctx ,
2007-05-14 13:01:28 +00:00
struct byte_range_lock * br_lck ,
2009-10-05 10:27:48 -07:00
struct lock_struct * plock )
2006-04-10 15:33:04 +00:00
{
2009-10-05 10:27:48 -07:00
unsigned int i , j , count ;
2006-04-10 15:33:04 +00:00
struct lock_struct * tp ;
2007-05-05 20:43:06 +00:00
struct lock_struct * locks = br_lck - > lock_data ;
2007-10-18 17:40:25 -07:00
bool overlap_found = False ;
2006-04-10 15:33:04 +00:00
/* No zero-zero locks for POSIX. */
if ( plock - > start = = 0 & & plock - > size = = 0 ) {
return False ;
}
2003-02-27 01:04:34 +00:00
2006-04-10 15:33:04 +00:00
/* Don't allow 64-bit lock wrap. */
if ( plock - > start + plock - > size < plock - > start | |
plock - > start + plock - > size < plock - > size ) {
DEBUG ( 10 , ( " brl_unlock_posix: lock wrap \n " ) ) ;
return False ;
}
2003-04-04 20:38:12 +00:00
2006-04-10 15:33:04 +00:00
/* The worst case scenario here is we have to split an
existing POSIX lock range into two , so we need at most
1 more entry . */
2003-04-04 20:38:12 +00:00
2013-09-10 21:04:47 +02:00
tp = talloc_array ( br_lck , struct lock_struct , br_lck - > num_locks + 1 ) ;
2006-04-10 15:33:04 +00:00
if ( ! tp ) {
DEBUG ( 10 , ( " brl_unlock_posix: malloc fail \n " ) ) ;
return False ;
}
2003-02-27 01:04:34 +00:00
2009-10-05 10:27:48 -07:00
count = 0 ;
2006-04-10 15:33:04 +00:00
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
2006-07-15 00:34:08 +00:00
struct lock_struct * lock = & locks [ i ] ;
2006-04-10 15:33:04 +00:00
unsigned int tmp_count ;
2003-02-27 01:04:34 +00:00
2006-04-10 15:33:04 +00:00
/* Only remove our own locks - ignore fnum. */
2006-07-29 19:14:24 +00:00
if ( IS_PENDING_LOCK ( lock - > lock_type ) | |
2006-04-10 15:33:04 +00:00
! brl_same_context ( & lock - > context , & plock - > context ) ) {
memcpy ( & tp [ count ] , lock , sizeof ( struct lock_struct ) ) ;
count + + ;
continue ;
}
2003-02-27 01:04:34 +00:00
2009-10-05 10:27:48 -07:00
if ( lock - > lock_flav = = WINDOWS_LOCK ) {
/* Do any Windows flavour locks conflict ? */
if ( brl_conflict ( lock , plock ) ) {
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( tp ) ;
2009-10-05 10:27:48 -07:00
return false ;
2006-04-10 15:33:04 +00:00
}
2009-10-05 10:27:48 -07:00
/* Just copy the Windows lock into the new array. */
memcpy ( & tp [ count ] , lock , sizeof ( struct lock_struct ) ) ;
2006-04-10 15:33:04 +00:00
count + + ;
continue ;
2009-10-05 10:27:48 -07:00
}
/* Work out overlaps. */
tmp_count = brlock_posix_split_merge ( & tp [ count ] , lock , plock ) ;
if ( tmp_count = = 0 ) {
/* plock overlapped the existing lock completely,
or replaced it . Don ' t copy the existing lock . */
overlap_found = true ;
} else if ( tmp_count = = 1 ) {
/* Either no overlap, (simple copy of existing lock) or
* an overlap of an existing lock . */
/* If the lock changed size, we had an overlap. */
if ( tp [ count ] . size ! = lock - > size ) {
overlap_found = true ;
}
count + = tmp_count ;
} else if ( tmp_count = = 2 ) {
/* We split a lock range in two. */
overlap_found = true ;
count + = tmp_count ;
2006-04-10 15:33:04 +00:00
/* Optimisation... */
/* We know we're finished here as we can't overlap any
more POSIX locks . Copy the rest of the lock array . */
2009-10-05 10:27:48 -07:00
2006-04-10 15:33:04 +00:00
if ( i < br_lck - > num_locks - 1 ) {
2009-10-05 10:27:48 -07:00
memcpy ( & tp [ count ] , & locks [ i + 1 ] ,
2006-04-10 15:33:04 +00:00
sizeof ( * locks ) * ( ( br_lck - > num_locks - 1 ) - i ) ) ;
count + = ( ( br_lck - > num_locks - 1 ) - i ) ;
}
break ;
}
2009-10-05 10:27:48 -07:00
2006-04-10 15:33:04 +00:00
}
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
if ( ! overlap_found ) {
/* Just ignore - no change. */
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( tp ) ;
2006-04-10 15:33:04 +00:00
DEBUG ( 10 , ( " brl_unlock_posix: No overlap - unlocked. \n " ) ) ;
return True ;
}
/* Unlock any POSIX regions. */
2006-11-11 17:05:11 +00:00
if ( lp_posix_locking ( br_lck - > fsp - > conn - > params ) ) {
2006-07-11 18:01:26 +00:00
release_posix_lock_posix_flavour ( br_lck - > fsp ,
plock - > start ,
plock - > size ,
& plock - > context ,
tp ,
count ) ;
2006-04-10 15:33:04 +00:00
}
/* Realloc so we don't leak entries per unlock call. */
if ( count ) {
2013-09-10 21:04:47 +02:00
tp = talloc_realloc ( br_lck , tp , struct lock_struct , count ) ;
2006-04-10 15:33:04 +00:00
if ( ! tp ) {
DEBUG ( 10 , ( " brl_unlock_posix: realloc fail \n " ) ) ;
return False ;
2000-01-13 12:09:36 +00:00
}
2006-04-10 15:33:04 +00:00
} else {
/* We deleted the last lock. */
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( tp ) ;
2006-04-10 15:33:04 +00:00
tp = NULL ;
2000-01-13 12:09:36 +00:00
}
2009-10-05 10:27:48 -07:00
contend_level2_oplocks_end ( br_lck - > fsp ,
LEVEL2_CONTEND_POSIX_BRL ) ;
2009-02-03 11:56:35 -08:00
2006-04-10 15:33:04 +00:00
br_lck - > num_locks = count ;
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( br_lck - > lock_data ) ;
2006-08-08 09:56:38 +00:00
locks = tp ;
2007-05-05 20:43:06 +00:00
br_lck - > lock_data = tp ;
2006-04-10 15:33:04 +00:00
br_lck - > modified = True ;
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
/* Send unlock messages to any pending waiters that overlap. */
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
for ( j = 0 ; j < br_lck - > num_locks ; j + + ) {
struct lock_struct * pend_lock = & locks [ j ] ;
/* Ignore non-pending locks. */
2006-07-29 19:14:24 +00:00
if ( ! IS_PENDING_LOCK ( pend_lock - > lock_type ) ) {
2006-04-10 15:33:04 +00:00
continue ;
}
/* We could send specific lock info here... */
2006-07-15 00:34:08 +00:00
if ( brl_pending_overlap ( plock , pend_lock ) ) {
2006-04-10 15:33:04 +00:00
DEBUG ( 10 , ( " brl_unlock: sending unlock message to pid %s \n " ,
procid_str_static ( & pend_lock - > context . pid ) ) ) ;
2007-05-14 13:01:28 +00:00
messaging_send ( msg_ctx , pend_lock - > context . pid ,
MSG_SMB_UNLOCK , & data_blob_null ) ;
2006-04-10 15:33:04 +00:00
}
}
return True ;
}
2001-06-30 01:59:48 +00:00
2009-07-23 20:28:58 -04:00
bool smb_vfs_call_brl_unlock_windows ( struct vfs_handle_struct * handle ,
struct messaging_context * msg_ctx ,
struct byte_range_lock * br_lck ,
const struct lock_struct * plock )
{
VFS_FIND ( brl_unlock_windows ) ;
2013-09-10 11:35:01 +00:00
return handle - > fns - > brl_unlock_windows_fn ( handle , msg_ctx , br_lck ,
2011-12-03 20:45:04 -08:00
plock ) ;
2009-07-23 20:28:58 -04:00
}
2000-01-13 12:09:36 +00:00
/****************************************************************************
2006-04-10 15:33:04 +00:00
Unlock a range of bytes .
2000-01-13 12:09:36 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-25 20:30:58 +00:00
2007-10-18 17:40:25 -07:00
bool brl_unlock ( struct messaging_context * msg_ctx ,
2007-05-14 13:01:28 +00:00
struct byte_range_lock * br_lck ,
2010-05-07 06:20:50 -07:00
uint64_t smblctx ,
2007-05-07 09:35:35 +00:00
struct server_id pid ,
2006-04-10 15:33:04 +00:00
br_off start ,
br_off size ,
enum brl_flavour lock_flav )
2000-01-13 12:09:36 +00:00
{
2006-04-10 15:33:04 +00:00
struct lock_struct lock ;
2000-01-13 12:09:36 +00:00
2010-05-07 06:20:50 -07:00
lock . context . smblctx = smblctx ;
2006-04-10 15:33:04 +00:00
lock . context . pid = pid ;
lock . context . tid = br_lck - > fsp - > conn - > cnum ;
lock . start = start ;
lock . size = size ;
lock . fnum = br_lck - > fsp - > fnum ;
lock . lock_type = UNLOCK_LOCK ;
lock . lock_flav = lock_flav ;
if ( lock_flav = = WINDOWS_LOCK ) {
2009-02-09 21:51:29 -08:00
return SMB_VFS_BRL_UNLOCK_WINDOWS ( br_lck - > fsp - > conn , msg_ctx ,
br_lck , & lock ) ;
2006-04-10 15:33:04 +00:00
} else {
2007-05-14 13:01:28 +00:00
return brl_unlock_posix ( msg_ctx , br_lck , & lock ) ;
2006-04-10 15:33:04 +00:00
}
}
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
/****************************************************************************
Test if we could add a lock if we wanted to .
Returns True if the region required is currently unlocked , False if locked .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-01-13 12:09:36 +00:00
2014-07-11 15:35:45 +02:00
bool brl_locktest ( const struct byte_range_lock * br_lck ,
const struct lock_struct * rw_probe )
2006-04-10 15:33:04 +00:00
{
2007-10-18 17:40:25 -07:00
bool ret = True ;
2006-04-10 15:33:04 +00:00
unsigned int i ;
2007-05-05 20:43:06 +00:00
const struct lock_struct * locks = br_lck - > lock_data ;
2006-04-10 15:33:04 +00:00
files_struct * fsp = br_lck - > fsp ;
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
/* Make sure existing locks don't conflict */
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
/*
* Our own locks don ' t conflict .
*/
2014-07-11 15:35:45 +02:00
if ( brl_conflict_other ( & locks [ i ] , rw_probe ) ) {
2006-04-10 15:33:04 +00:00
return False ;
2000-01-13 12:09:36 +00:00
}
}
2006-04-10 15:33:04 +00:00
/*
* There is no lock held by an SMB daemon , check to
* see if there is a POSIX lock from a UNIX or NFS process .
* This only conflicts with Windows locks , not POSIX locks .
*/
2014-07-11 15:35:45 +02:00
if ( lp_posix_locking ( fsp - > conn - > params ) & &
( rw_probe - > lock_flav = = WINDOWS_LOCK ) ) {
/*
* Make copies - - is_posix_locked might modify the values
*/
br_off start = rw_probe - > start ;
br_off size = rw_probe - > size ;
enum brl_type lock_type = rw_probe - > lock_type ;
2006-04-10 15:33:04 +00:00
ret = is_posix_locked ( fsp , & start , & size , & lock_type , WINDOWS_LOCK ) ;
2014-07-11 12:52:06 +02:00
DEBUG ( 10 , ( " brl_locktest: posix start=%ju len=%ju %s for %s "
" file %s \n " , ( uintmax_t ) start , ( uintmax_t ) size ,
ret ? " locked " : " unlocked " ,
fsp_fnum_dbg ( fsp ) , fsp_str_dbg ( fsp ) ) ) ;
2006-04-10 15:33:04 +00:00
/* We need to return the inverse of is_posix_locked. */
ret = ! ret ;
}
2000-01-13 12:09:36 +00:00
/* no conflicts - we could have added it */
2006-04-10 15:33:04 +00:00
return ret ;
}
2000-01-13 12:09:36 +00:00
2006-04-10 15:33:04 +00:00
/****************************************************************************
Query for existing locks .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
NTSTATUS brl_lockquery ( struct byte_range_lock * br_lck ,
2010-05-07 06:20:50 -07:00
uint64_t * psmblctx ,
2007-05-07 09:35:35 +00:00
struct server_id pid ,
2006-04-10 15:33:04 +00:00
br_off * pstart ,
2013-09-10 11:35:01 +00:00
br_off * psize ,
2006-04-10 15:33:04 +00:00
enum brl_type * plock_type ,
enum brl_flavour lock_flav )
{
unsigned int i ;
struct lock_struct lock ;
2007-05-05 20:43:06 +00:00
const struct lock_struct * locks = br_lck - > lock_data ;
2006-04-10 15:33:04 +00:00
files_struct * fsp = br_lck - > fsp ;
2010-05-07 06:20:50 -07:00
lock . context . smblctx = * psmblctx ;
2006-04-10 15:33:04 +00:00
lock . context . pid = pid ;
lock . context . tid = br_lck - > fsp - > conn - > cnum ;
lock . start = * pstart ;
lock . size = * psize ;
lock . fnum = fsp - > fnum ;
lock . lock_type = * plock_type ;
lock . lock_flav = lock_flav ;
/* Make sure existing locks don't conflict */
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
2006-07-15 00:05:47 +00:00
const struct lock_struct * exlock = & locks [ i ] ;
2007-10-18 17:40:25 -07:00
bool conflict = False ;
2006-04-10 15:33:04 +00:00
if ( exlock - > lock_flav = = WINDOWS_LOCK ) {
conflict = brl_conflict ( exlock , & lock ) ;
2013-09-10 11:35:01 +00:00
} else {
2006-04-10 15:33:04 +00:00
conflict = brl_conflict_posix ( exlock , & lock ) ;
}
if ( conflict ) {
2010-05-07 06:20:50 -07:00
* psmblctx = exlock - > context . smblctx ;
2006-04-10 15:33:04 +00:00
* pstart = exlock - > start ;
* psize = exlock - > size ;
* plock_type = exlock - > lock_type ;
return NT_STATUS_LOCK_NOT_GRANTED ;
}
}
/*
* There is no lock held by an SMB daemon , check to
* see if there is a POSIX lock from a UNIX or NFS process .
*/
2006-11-11 17:05:11 +00:00
if ( lp_posix_locking ( fsp - > conn - > params ) ) {
2007-10-18 17:40:25 -07:00
bool ret = is_posix_locked ( fsp , pstart , psize , plock_type , POSIX_LOCK ) ;
2006-04-10 15:33:04 +00:00
2014-07-11 12:52:06 +02:00
DEBUG ( 10 , ( " brl_lockquery: posix start=%ju len=%ju %s for %s "
" file %s \n " , ( uintmax_t ) * pstart ,
( uintmax_t ) * psize , ret ? " locked " : " unlocked " ,
fsp_fnum_dbg ( fsp ) , fsp_str_dbg ( fsp ) ) ) ;
2006-04-10 15:33:04 +00:00
if ( ret ) {
2010-05-07 06:20:50 -07:00
/* Hmmm. No clue what to set smblctx to - use -1. */
* psmblctx = 0xFFFFFFFFFFFFFFFFLL ;
2006-04-10 15:33:04 +00:00
return NT_STATUS_LOCK_NOT_GRANTED ;
}
}
return NT_STATUS_OK ;
2000-01-13 12:09:36 +00:00
}
2000-01-14 04:32:57 +00:00
2009-07-23 20:28:58 -04:00
bool smb_vfs_call_brl_cancel_windows ( struct vfs_handle_struct * handle ,
struct byte_range_lock * br_lck ,
2014-07-03 13:51:45 +00:00
struct lock_struct * plock )
2009-07-23 20:28:58 -04:00
{
VFS_FIND ( brl_cancel_windows ) ;
2014-07-03 13:51:45 +00:00
return handle - > fns - > brl_cancel_windows_fn ( handle , br_lck , plock ) ;
2009-07-23 20:28:58 -04:00
}
2000-01-14 04:32:57 +00:00
/****************************************************************************
2006-04-10 15:33:04 +00:00
Remove a particular pending lock .
2000-01-14 04:32:57 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
bool brl_lock_cancel ( struct byte_range_lock * br_lck ,
2010-05-07 06:20:50 -07:00
uint64_t smblctx ,
2007-05-07 09:35:35 +00:00
struct server_id pid ,
2006-04-10 15:33:04 +00:00
br_off start ,
br_off size ,
2014-07-03 14:03:12 +00:00
enum brl_flavour lock_flav )
2009-02-09 21:51:29 -08:00
{
bool ret ;
struct lock_struct lock ;
2010-05-07 06:20:50 -07:00
lock . context . smblctx = smblctx ;
2009-02-09 21:51:29 -08:00
lock . context . pid = pid ;
lock . context . tid = br_lck - > fsp - > conn - > cnum ;
lock . start = start ;
lock . size = size ;
lock . fnum = br_lck - > fsp - > fnum ;
lock . lock_flav = lock_flav ;
/* lock.lock_type doesn't matter */
if ( lock_flav = = WINDOWS_LOCK ) {
ret = SMB_VFS_BRL_CANCEL_WINDOWS ( br_lck - > fsp - > conn , br_lck ,
2014-07-03 13:51:45 +00:00
& lock ) ;
2009-02-09 21:51:29 -08:00
} else {
ret = brl_lock_cancel_default ( br_lck , & lock ) ;
}
return ret ;
}
bool brl_lock_cancel_default ( struct byte_range_lock * br_lck ,
struct lock_struct * plock )
2000-01-14 04:32:57 +00:00
{
2006-04-10 15:33:04 +00:00
unsigned int i ;
2007-05-05 20:43:06 +00:00
struct lock_struct * locks = br_lck - > lock_data ;
2006-04-10 15:33:04 +00:00
2009-02-09 21:51:29 -08:00
SMB_ASSERT ( plock ) ;
2006-04-10 15:33:04 +00:00
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
struct lock_struct * lock = & locks [ i ] ;
/* For pending locks we *always* care about the fnum. */
2009-02-09 21:51:29 -08:00
if ( brl_same_context ( & lock - > context , & plock - > context ) & &
lock - > fnum = = plock - > fnum & &
2006-07-29 19:14:24 +00:00
IS_PENDING_LOCK ( lock - > lock_type ) & &
2009-02-09 21:51:29 -08:00
lock - > lock_flav = = plock - > lock_flav & &
lock - > start = = plock - > start & &
lock - > size = = plock - > size ) {
2006-04-10 15:33:04 +00:00
break ;
}
}
2000-01-14 04:32:57 +00:00
2006-04-10 15:33:04 +00:00
if ( i = = br_lck - > num_locks ) {
/* Didn't find it. */
return False ;
}
2000-01-14 04:32:57 +00:00
2014-07-03 10:18:36 +00:00
brl_delete_lock_struct ( locks , br_lck - > num_locks , i ) ;
2006-04-10 15:33:04 +00:00
br_lck - > num_locks - = 1 ;
br_lck - > modified = True ;
return True ;
}
2000-01-14 04:32:57 +00:00
2006-04-10 15:33:04 +00:00
/****************************************************************************
Remove any locks associated with a open file .
2006-07-11 18:01:26 +00:00
We return True if this process owns any other Windows locks on this
fd and so we should not immediately close the fd .
2006-04-10 15:33:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-05-14 13:01:28 +00:00
void brl_close_fnum ( struct messaging_context * msg_ctx ,
struct byte_range_lock * br_lck )
2006-04-10 15:33:04 +00:00
{
files_struct * fsp = br_lck - > fsp ;
2012-06-06 15:28:14 +02:00
uint32_t tid = fsp - > conn - > cnum ;
2012-06-28 09:54:41 +02:00
uint64_t fnum = fsp - > fnum ;
2011-07-15 16:11:07 -07:00
unsigned int i ;
2007-05-05 20:43:06 +00:00
struct lock_struct * locks = br_lck - > lock_data ;
2011-12-15 11:50:43 +01:00
struct server_id pid = messaging_server_id ( fsp - > conn - > sconn - > msg_ctx ) ;
2011-07-15 16:11:07 -07:00
struct lock_struct * locks_copy ;
unsigned int num_locks_copy ;
/* Copy the current lock array. */
if ( br_lck - > num_locks ) {
locks_copy = ( struct lock_struct * ) talloc_memdup ( br_lck , locks , br_lck - > num_locks * sizeof ( struct lock_struct ) ) ;
if ( ! locks_copy ) {
smb_panic ( " brl_close_fnum: talloc failed " ) ;
2006-07-11 18:01:26 +00:00
}
2012-09-04 11:56:15 +02:00
} else {
2011-07-15 16:11:07 -07:00
locks_copy = NULL ;
2006-07-11 18:01:26 +00:00
}
2011-07-15 16:11:07 -07:00
num_locks_copy = br_lck - > num_locks ;
2006-07-11 18:01:26 +00:00
2011-07-15 16:11:07 -07:00
for ( i = 0 ; i < num_locks_copy ; i + + ) {
struct lock_struct * lock = & locks_copy [ i ] ;
2003-02-27 01:04:34 +00:00
2012-06-16 00:26:26 +02:00
if ( lock - > context . tid = = tid & & serverid_equal ( & lock - > context . pid , & pid ) & &
2011-07-15 16:11:07 -07:00
( lock - > fnum = = fnum ) ) {
brl_unlock ( msg_ctx ,
br_lck ,
lock - > context . smblctx ,
pid ,
lock - > start ,
lock - > size ,
lock - > lock_flav ) ;
2006-04-10 15:33:04 +00:00
}
2006-07-12 16:32:02 +00:00
}
2000-01-14 04:32:57 +00:00
}
2000-01-16 11:14:44 +00:00
2012-06-30 21:48:43 +02:00
bool brl_mark_disconnected ( struct files_struct * fsp )
{
uint32_t tid = fsp - > conn - > cnum ;
2014-05-01 10:58:51 -07:00
uint64_t smblctx ;
2012-06-30 21:48:43 +02:00
uint64_t fnum = fsp - > fnum ;
unsigned int i ;
struct server_id self = messaging_server_id ( fsp - > conn - > sconn - > msg_ctx ) ;
struct byte_range_lock * br_lck = NULL ;
2014-05-01 10:58:51 -07:00
if ( fsp - > op = = NULL ) {
return false ;
}
smblctx = fsp - > op - > global - > open_persistent_id ;
2012-06-30 21:48:43 +02:00
if ( ! fsp - > op - > global - > durable ) {
return false ;
}
if ( fsp - > current_lock_count = = 0 ) {
return true ;
}
br_lck = brl_get_locks ( talloc_tos ( ) , fsp ) ;
if ( br_lck = = NULL ) {
return false ;
}
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
struct lock_struct * lock = & br_lck - > lock_data [ i ] ;
/*
* as this is a durable handle , we only expect locks
* of the current file handle !
*/
if ( lock - > context . smblctx ! = smblctx ) {
TALLOC_FREE ( br_lck ) ;
return false ;
}
if ( lock - > context . tid ! = tid ) {
TALLOC_FREE ( br_lck ) ;
return false ;
}
if ( ! serverid_equal ( & lock - > context . pid , & self ) ) {
TALLOC_FREE ( br_lck ) ;
return false ;
}
if ( lock - > fnum ! = fnum ) {
TALLOC_FREE ( br_lck ) ;
return false ;
}
server_id_set_disconnected ( & lock - > context . pid ) ;
lock - > context . tid = TID_FIELD_INVALID ;
lock - > fnum = FNUM_FIELD_INVALID ;
}
br_lck - > modified = true ;
TALLOC_FREE ( br_lck ) ;
return true ;
}
bool brl_reconnect_disconnected ( struct files_struct * fsp )
{
uint32_t tid = fsp - > conn - > cnum ;
2014-05-01 10:58:51 -07:00
uint64_t smblctx ;
2012-06-30 21:48:43 +02:00
uint64_t fnum = fsp - > fnum ;
unsigned int i ;
struct server_id self = messaging_server_id ( fsp - > conn - > sconn - > msg_ctx ) ;
struct byte_range_lock * br_lck = NULL ;
2014-05-01 10:58:51 -07:00
if ( fsp - > op = = NULL ) {
return false ;
}
smblctx = fsp - > op - > global - > open_persistent_id ;
2012-06-30 21:48:43 +02:00
if ( ! fsp - > op - > global - > durable ) {
return false ;
}
2013-04-12 11:13:57 +02:00
/*
* When reconnecting , we do not want to validate the brlock entries
* and thereby remove our own ( disconnected ) entries but reactivate
* them instead .
*/
2012-06-30 21:48:43 +02:00
fsp - > lockdb_clean = true ;
br_lck = brl_get_locks ( talloc_tos ( ) , fsp ) ;
if ( br_lck = = NULL ) {
return false ;
}
if ( br_lck - > num_locks = = 0 ) {
TALLOC_FREE ( br_lck ) ;
return true ;
}
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
struct lock_struct * lock = & br_lck - > lock_data [ i ] ;
/*
* as this is a durable handle we only expect locks
* of the current file handle !
*/
if ( lock - > context . smblctx ! = smblctx ) {
TALLOC_FREE ( br_lck ) ;
return false ;
}
if ( lock - > context . tid ! = TID_FIELD_INVALID ) {
TALLOC_FREE ( br_lck ) ;
return false ;
}
if ( ! server_id_is_disconnected ( & lock - > context . pid ) ) {
TALLOC_FREE ( br_lck ) ;
return false ;
}
if ( lock - > fnum ! = FNUM_FIELD_INVALID ) {
TALLOC_FREE ( br_lck ) ;
return false ;
}
lock - > context . pid = self ;
lock - > context . tid = tid ;
lock - > fnum = fnum ;
}
fsp - > current_lock_count = br_lck - > num_locks ;
br_lck - > modified = true ;
TALLOC_FREE ( br_lck ) ;
return true ;
}
2000-01-16 11:14:44 +00:00
/****************************************************************************
2006-05-03 16:07:21 +00:00
Ensure this set of lock entries is valid .
2000-01-16 11:14:44 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2014-06-29 11:38:35 +02:00
static bool validate_lock_entries ( unsigned int * pnum_entries , struct lock_struct * * pplocks ,
2013-03-05 14:49:28 +01:00
bool keep_disconnected )
2000-01-16 11:14:44 +00:00
{
2006-05-03 02:14:09 +00:00
unsigned int i ;
2006-05-03 16:07:21 +00:00
struct lock_struct * locks = * pplocks ;
2014-06-29 11:22:13 +02:00
unsigned int num_entries = * pnum_entries ;
2013-12-06 10:31:22 +00:00
TALLOC_CTX * frame ;
2013-03-05 14:02:10 +01:00
struct server_id * ids ;
bool * exists ;
2014-06-29 11:22:13 +02:00
if ( num_entries = = 0 ) {
2013-12-06 10:31:22 +00:00
return true ;
}
frame = talloc_stackframe ( ) ;
2014-06-29 11:22:13 +02:00
ids = talloc_array ( frame , struct server_id , num_entries ) ;
2013-03-05 14:02:10 +01:00
if ( ids = = NULL ) {
DEBUG ( 0 , ( " validate_lock_entries: "
" talloc_array(struct server_id, %u) failed \n " ,
2014-06-29 11:22:13 +02:00
num_entries ) ) ;
2013-03-05 14:02:10 +01:00
talloc_free ( frame ) ;
return false ;
}
2014-06-29 11:22:13 +02:00
exists = talloc_array ( frame , bool , num_entries ) ;
2013-03-05 14:02:10 +01:00
if ( exists = = NULL ) {
DEBUG ( 0 , ( " validate_lock_entries: "
" talloc_array(bool, %u) failed \n " ,
2014-06-29 11:22:13 +02:00
num_entries ) ) ;
2013-03-05 14:02:10 +01:00
talloc_free ( frame ) ;
return false ;
}
2014-06-29 11:22:13 +02:00
for ( i = 0 ; i < num_entries ; i + + ) {
2013-03-05 14:02:10 +01:00
ids [ i ] = locks [ i ] . context . pid ;
}
2014-06-29 11:22:13 +02:00
if ( ! serverids_exist ( ids , num_entries , exists ) ) {
2013-03-05 14:02:10 +01:00
DEBUG ( 3 , ( " validate_lock_entries: serverids_exists failed \n " ) ) ;
talloc_free ( frame ) ;
return false ;
}
2000-01-16 11:14:44 +00:00
2014-06-29 11:36:24 +02:00
i = 0 ;
while ( i < num_entries ) {
2013-03-05 14:49:28 +01:00
if ( exists [ i ] ) {
2014-06-29 11:36:24 +02:00
i + + ;
2013-03-05 14:49:28 +01:00
continue ;
2006-05-03 02:14:09 +00:00
}
2013-03-05 14:49:28 +01:00
if ( keep_disconnected & &
server_id_is_disconnected ( & ids [ i ] ) )
{
2014-06-29 11:36:24 +02:00
i + + ;
2013-03-05 14:49:28 +01:00
continue ;
}
2014-06-29 11:36:24 +02:00
/* This process no longer exists */
brl_delete_lock_struct ( locks , num_entries , i ) ;
num_entries - = 1 ;
2006-05-03 02:14:09 +00:00
}
2013-03-05 14:02:10 +01:00
TALLOC_FREE ( frame ) ;
2006-05-03 02:14:09 +00:00
2014-06-29 11:36:24 +02:00
* pnum_entries = num_entries ;
2006-05-03 16:07:21 +00:00
return True ;
}
2007-05-29 13:26:44 +00:00
struct brl_forall_cb {
void ( * fn ) ( struct file_id id , struct server_id pid ,
enum brl_type lock_type ,
enum brl_flavour lock_flav ,
br_off start , br_off size ,
void * private_data ) ;
void * private_data ;
} ;
2006-05-03 16:07:21 +00:00
/****************************************************************************
Traverse the whole database with this function , calling traverse_callback
on each lock .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2012-08-08 17:46:59 +02:00
static int brl_traverse_fn ( struct db_record * rec , void * state )
2006-05-03 16:07:21 +00:00
{
2007-05-29 13:26:44 +00:00
struct brl_forall_cb * cb = ( struct brl_forall_cb * ) state ;
2006-05-03 16:07:21 +00:00
struct lock_struct * locks ;
2007-05-29 09:30:34 +00:00
struct file_id * key ;
2006-05-03 16:07:21 +00:00
unsigned int i ;
unsigned int num_locks = 0 ;
unsigned int orig_num_locks = 0 ;
2011-08-17 10:53:58 +02:00
TDB_DATA dbkey ;
TDB_DATA value ;
dbkey = dbwrap_record_get_key ( rec ) ;
value = dbwrap_record_get_value ( rec ) ;
2006-05-03 16:07:21 +00:00
/* In a traverse function we must make a copy of
dbuf before modifying it . */
2013-09-10 21:04:47 +02:00
locks = ( struct lock_struct * ) talloc_memdup (
talloc_tos ( ) , value . dptr , value . dsize ) ;
2006-05-03 16:07:21 +00:00
if ( ! locks ) {
return - 1 ; /* Terminate traversal. */
}
2011-08-17 10:53:58 +02:00
key = ( struct file_id * ) dbkey . dptr ;
orig_num_locks = num_locks = value . dsize / sizeof ( * locks ) ;
2006-05-03 16:07:21 +00:00
/* Ensure the lock db is clean of entries from invalid processes. */
2014-06-29 11:38:35 +02:00
if ( ! validate_lock_entries ( & num_locks , & locks , true ) ) {
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( locks ) ;
2006-05-03 16:07:21 +00:00
return - 1 ; /* Terminate traversal */
}
if ( orig_num_locks ! = num_locks ) {
2007-05-27 17:12:08 +00:00
if ( num_locks ) {
TDB_DATA data ;
data . dptr = ( uint8_t * ) locks ;
data . dsize = num_locks * sizeof ( struct lock_struct ) ;
2011-08-17 10:53:58 +02:00
dbwrap_record_store ( rec , data , TDB_REPLACE ) ;
2006-05-03 02:14:09 +00:00
} else {
2011-08-17 10:53:58 +02:00
dbwrap_record_delete ( rec ) ;
2006-05-03 02:14:09 +00:00
}
}
2008-01-16 12:09:48 +03:00
if ( cb - > fn ) {
for ( i = 0 ; i < num_locks ; i + + ) {
cb - > fn ( * key ,
locks [ i ] . context . pid ,
locks [ i ] . lock_type ,
locks [ i ] . lock_flav ,
locks [ i ] . start ,
locks [ i ] . size ,
cb - > private_data ) ;
}
2000-01-16 11:14:44 +00:00
}
2006-05-03 16:07:21 +00:00
2013-09-10 21:04:47 +02:00
TALLOC_FREE ( locks ) ;
2000-01-16 11:14:44 +00:00
return 0 ;
}
/*******************************************************************
2000-04-25 20:30:58 +00:00
Call the specified function on each lock in the database .
2000-01-16 11:14:44 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-25 20:30:58 +00:00
2007-05-29 13:26:44 +00:00
int brl_forall ( void ( * fn ) ( struct file_id id , struct server_id pid ,
enum brl_type lock_type ,
enum brl_flavour lock_flav ,
br_off start , br_off size ,
void * private_data ) ,
void * private_data )
2000-01-16 11:14:44 +00:00
{
2007-05-29 13:26:44 +00:00
struct brl_forall_cb cb ;
2011-08-17 10:53:58 +02:00
NTSTATUS status ;
int count = 0 ;
2007-05-29 13:26:44 +00:00
2007-05-27 10:35:14 +00:00
if ( ! brlock_db ) {
2006-04-10 15:33:04 +00:00
return 0 ;
}
2007-05-29 13:26:44 +00:00
cb . fn = fn ;
cb . private_data = private_data ;
2012-08-08 17:46:59 +02:00
status = dbwrap_traverse ( brlock_db , brl_traverse_fn , & cb , & count ) ;
2011-08-17 10:53:58 +02:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return - 1 ;
} else {
return count ;
}
2000-01-16 11:14:44 +00:00
}
2006-04-10 15:33:04 +00:00
/*******************************************************************
Store a potentially modified set of byte range lock data back into
the database .
Unlock the record .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2011-03-17 10:04:12 +01:00
static void byte_range_lock_flush ( struct byte_range_lock * br_lck )
2006-04-10 15:33:04 +00:00
{
2013-09-11 12:48:14 +00:00
size_t data_len ;
2006-04-10 15:33:04 +00:00
if ( ! br_lck - > modified ) {
2013-09-13 14:13:51 +02:00
DEBUG ( 10 , ( " br_lck not modified \n " ) ) ;
2006-04-10 15:33:04 +00:00
goto done ;
}
2013-09-11 12:48:14 +00:00
data_len = br_lck - > num_locks * sizeof ( struct lock_struct ) ;
if ( br_lck - > have_read_oplocks ) {
data_len + = 1 ;
}
2013-09-13 14:13:51 +02:00
DEBUG ( 10 , ( " data_len=%d \n " , ( int ) data_len ) ) ;
2013-09-11 12:48:14 +00:00
if ( data_len = = 0 ) {
2006-04-10 15:33:04 +00:00
/* No locks - delete this entry. */
2011-08-17 10:53:58 +02:00
NTSTATUS status = dbwrap_record_delete ( br_lck - > record ) ;
2007-05-27 10:35:14 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " delete_rec returned %s \n " ,
nt_errstr ( status ) ) ) ;
2007-06-15 21:58:49 +00:00
smb_panic ( " Could not delete byte range lock entry " ) ;
2006-04-10 15:33:04 +00:00
}
} else {
TDB_DATA data ;
2007-05-27 10:35:14 +00:00
NTSTATUS status ;
2013-09-11 12:48:14 +00:00
data . dsize = data_len ;
data . dptr = talloc_array ( talloc_tos ( ) , uint8_t , data_len ) ;
SMB_ASSERT ( data . dptr ! = NULL ) ;
memcpy ( data . dptr , br_lck - > lock_data ,
br_lck - > num_locks * sizeof ( struct lock_struct ) ) ;
if ( br_lck - > have_read_oplocks ) {
data . dptr [ data_len - 1 ] = 1 ;
}
2006-04-10 15:33:04 +00:00
2011-08-17 10:53:58 +02:00
status = dbwrap_record_store ( br_lck - > record , data , TDB_REPLACE ) ;
2013-09-11 12:48:14 +00:00
TALLOC_FREE ( data . dptr ) ;
2007-05-27 10:35:14 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " store returned %s \n " , nt_errstr ( status ) ) ) ;
2007-06-15 21:58:49 +00:00
smb_panic ( " Could not store byte range mode entry " ) ;
2006-04-10 15:33:04 +00:00
}
}
2013-09-13 14:13:51 +02:00
DEBUG ( 10 , ( " seqnum=%d \n " , dbwrap_get_seqnum ( brlock_db ) ) ) ;
2006-04-10 15:33:04 +00:00
done :
2011-03-17 10:04:12 +01:00
br_lck - > modified = false ;
2007-05-27 10:35:14 +00:00
TALLOC_FREE ( br_lck - > record ) ;
2011-03-17 10:04:12 +01:00
}
static int byte_range_lock_destructor ( struct byte_range_lock * br_lck )
{
byte_range_lock_flush ( br_lck ) ;
2006-04-10 15:33:04 +00:00
return 0 ;
}
/*******************************************************************
Fetch a set of byte range lock data from the database .
Leave the record locked .
2006-07-11 18:01:26 +00:00
TALLOC_FREE ( brl ) will release the lock in the destructor .
2006-04-10 15:33:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2013-09-11 11:53:26 +00:00
struct byte_range_lock * brl_get_locks ( TALLOC_CTX * mem_ctx , files_struct * fsp )
2006-04-10 15:33:04 +00:00
{
2007-05-27 10:35:14 +00:00
TDB_DATA key , data ;
2011-06-07 11:38:41 +10:00
struct byte_range_lock * br_lck = talloc ( mem_ctx , struct byte_range_lock ) ;
2006-04-10 15:33:04 +00:00
if ( br_lck = = NULL ) {
return NULL ;
}
br_lck - > fsp = fsp ;
br_lck - > num_locks = 0 ;
2013-09-11 12:48:14 +00:00
br_lck - > have_read_oplocks = false ;
2006-04-10 15:33:04 +00:00
br_lck - > modified = False ;
2006-04-12 23:00:58 +00:00
2013-09-11 09:31:36 +00:00
key . dptr = ( uint8 * ) & fsp - > file_id ;
2007-05-29 09:30:34 +00:00
key . dsize = sizeof ( struct file_id ) ;
2006-04-10 15:33:04 +00:00
2013-09-11 11:51:44 +00:00
br_lck - > record = dbwrap_fetch_locked ( brlock_db , br_lck , key ) ;
2007-05-27 10:35:14 +00:00
2013-09-11 11:51:44 +00:00
if ( br_lck - > record = = NULL ) {
DEBUG ( 3 , ( " Could not lock byte range lock entry \n " ) ) ;
TALLOC_FREE ( br_lck ) ;
return NULL ;
2006-04-10 15:33:04 +00:00
}
2013-09-11 11:51:44 +00:00
data = dbwrap_record_get_value ( br_lck - > record ) ;
2007-07-16 08:23:20 +00:00
br_lck - > lock_data = NULL ;
2007-05-27 10:35:14 +00:00
2006-07-11 18:01:26 +00:00
talloc_set_destructor ( br_lck , byte_range_lock_destructor ) ;
2006-04-10 15:33:04 +00:00
br_lck - > num_locks = data . dsize / sizeof ( struct lock_struct ) ;
2007-07-09 07:51:39 +00:00
2007-07-16 07:40:30 +00:00
if ( br_lck - > num_locks ! = 0 ) {
2013-09-10 21:04:47 +02:00
br_lck - > lock_data = talloc_array (
br_lck , struct lock_struct , br_lck - > num_locks ) ;
2007-07-16 07:40:30 +00:00
if ( br_lck - > lock_data = = NULL ) {
DEBUG ( 0 , ( " malloc failed \n " ) ) ;
TALLOC_FREE ( br_lck ) ;
return NULL ;
}
2013-09-11 12:48:14 +00:00
memcpy ( br_lck - > lock_data , data . dptr ,
talloc_get_size ( br_lck - > lock_data ) ) ;
}
2013-09-13 14:13:51 +02:00
DEBUG ( 10 , ( " data.dsize=%d \n " , ( int ) data . dsize ) ) ;
2013-09-11 12:48:14 +00:00
if ( ( data . dsize % sizeof ( struct lock_struct ) ) = = 1 ) {
br_lck - > have_read_oplocks = ( data . dptr [ data . dsize - 1 ] = = 1 ) ;
2007-07-16 07:40:30 +00:00
}
2011-05-28 10:24:20 +02:00
2006-05-03 02:14:09 +00:00
if ( ! fsp - > lockdb_clean ) {
2007-06-16 01:04:22 +00:00
int orig_num_locks = br_lck - > num_locks ;
2006-05-03 02:14:09 +00:00
2013-04-12 11:05:29 +02:00
/*
* This is the first time we access the byte range lock
* record with this fsp . Go through and ensure all entries
* are valid - remove any that don ' t .
* This makes the lockdb self cleaning at low cost .
2013-03-05 14:49:28 +01:00
*
* Note : Disconnected entries belong to disconnected
* durable handles . So at this point , we have a new
* handle on the file and the disconnected durable has
* already been closed ( we are not a durable reconnect ) .
* So we need to clean the disconnected brl entry .
2013-04-12 11:05:29 +02:00
*/
2006-05-03 02:14:09 +00:00
2014-06-29 11:38:35 +02:00
if ( ! validate_lock_entries ( & br_lck - > num_locks ,
2013-03-05 14:49:28 +01:00
& br_lck - > lock_data , false ) ) {
2006-07-11 18:01:26 +00:00
TALLOC_FREE ( br_lck ) ;
2006-05-03 16:07:21 +00:00
return NULL ;
2006-05-03 02:14:09 +00:00
}
2007-06-16 01:04:22 +00:00
/* Ensure invalid locks are cleaned up in the destructor. */
if ( orig_num_locks ! = br_lck - > num_locks ) {
br_lck - > modified = True ;
}
2006-05-03 02:14:09 +00:00
/* Mark the lockdb as "clean" as seen from this open file. */
fsp - > lockdb_clean = True ;
}
2006-04-10 15:33:04 +00:00
if ( DEBUGLEVEL > = 10 ) {
unsigned int i ;
2007-05-05 20:43:06 +00:00
struct lock_struct * locks = br_lck - > lock_data ;
2007-05-29 09:30:34 +00:00
DEBUG ( 10 , ( " brl_get_locks_internal: %u current locks on file_id %s \n " ,
2006-04-10 15:33:04 +00:00
br_lck - > num_locks ,
2007-09-10 10:56:07 +00:00
file_id_string_tos ( & fsp - > file_id ) ) ) ;
2006-04-10 15:33:04 +00:00
for ( i = 0 ; i < br_lck - > num_locks ; i + + ) {
print_lock_struct ( i , & locks [ i ] ) ;
}
}
2011-03-17 10:04:12 +01:00
2006-04-10 15:33:04 +00:00
return br_lck ;
}
2006-07-31 20:58:02 +00:00
2013-09-11 11:36:54 +00:00
struct brl_get_locks_readonly_state {
TALLOC_CTX * mem_ctx ;
struct byte_range_lock * * br_lock ;
} ;
static void brl_get_locks_readonly_parser ( TDB_DATA key , TDB_DATA data ,
void * private_data )
2006-07-31 20:58:02 +00:00
{
2013-09-11 11:36:54 +00:00
struct brl_get_locks_readonly_state * state =
( struct brl_get_locks_readonly_state * ) private_data ;
2009-11-16 09:40:47 +01:00
struct byte_range_lock * br_lock ;
2013-09-11 11:36:54 +00:00
br_lock = talloc_pooled_object (
state - > mem_ctx , struct byte_range_lock , 1 , data . dsize ) ;
if ( br_lock = = NULL ) {
* state - > br_lock = NULL ;
return ;
}
br_lock - > lock_data = ( struct lock_struct * ) talloc_memdup (
br_lock , data . dptr , data . dsize ) ;
br_lock - > num_locks = data . dsize / sizeof ( struct lock_struct ) ;
2013-09-11 12:48:14 +00:00
if ( ( data . dsize % sizeof ( struct lock_struct ) ) = = 1 ) {
br_lock - > have_read_oplocks = ( data . dptr [ data . dsize - 1 ] = = 1 ) ;
2014-03-03 13:49:46 +01:00
} else {
br_lock - > have_read_oplocks = false ;
2013-09-11 12:48:14 +00:00
}
2013-09-13 14:13:51 +02:00
DEBUG ( 10 , ( " Got %d bytes, have_read_oplocks: %s \n " , ( int ) data . dsize ,
br_lock - > have_read_oplocks ? " true " : " false " ) ) ;
2013-09-11 11:36:54 +00:00
* state - > br_lock = br_lock ;
}
struct byte_range_lock * brl_get_locks_readonly ( files_struct * fsp )
{
struct byte_range_lock * br_lock = NULL ;
struct byte_range_lock * rw = NULL ;
2013-09-13 14:13:51 +02:00
DEBUG ( 10 , ( " seqnum=%d, fsp->brlock_seqnum=%d \n " ,
dbwrap_get_seqnum ( brlock_db ) , fsp - > brlock_seqnum ) ) ;
2009-11-16 09:40:47 +01:00
if ( ( fsp - > brlock_rec ! = NULL )
2011-08-17 10:53:58 +02:00
& & ( dbwrap_get_seqnum ( brlock_db ) = = fsp - > brlock_seqnum ) ) {
2013-09-11 11:36:54 +00:00
/*
* We have cached the brlock_rec and the database did not
* change .
*/
2009-11-16 09:40:47 +01:00
return fsp - > brlock_rec ;
}
2013-09-11 11:36:54 +00:00
if ( ! fsp - > lockdb_clean ) {
/*
* Fetch the record in R / W mode to give validate_lock_entries
* a chance to kick in once .
*/
2013-09-11 11:53:26 +00:00
rw = brl_get_locks ( talloc_tos ( ) , fsp ) ;
2013-09-11 11:36:54 +00:00
if ( rw = = NULL ) {
return NULL ;
}
fsp - > lockdb_clean = true ;
2013-09-11 10:17:05 +00:00
}
2013-09-11 11:36:54 +00:00
if ( rw ! = NULL ) {
size_t lock_data_size ;
2009-11-16 09:40:47 +01:00
2013-09-11 11:36:54 +00:00
/*
* Make a copy of the already retrieved and sanitized rw record
*/
lock_data_size = rw - > num_locks * sizeof ( struct lock_struct ) ;
br_lock = talloc_pooled_object (
fsp , struct byte_range_lock , 1 , lock_data_size ) ;
if ( br_lock = = NULL ) {
goto fail ;
}
2013-09-11 12:48:14 +00:00
br_lock - > have_read_oplocks = rw - > have_read_oplocks ;
2013-09-11 11:36:54 +00:00
br_lock - > num_locks = rw - > num_locks ;
br_lock - > lock_data = ( struct lock_struct * ) talloc_memdup (
br_lock , rw - > lock_data , lock_data_size ) ;
} else {
struct brl_get_locks_readonly_state state ;
NTSTATUS status ;
/*
* Parse the record fresh from the database
*/
state . mem_ctx = fsp ;
state . br_lock = & br_lock ;
status = dbwrap_parse_record (
brlock_db ,
make_tdb_data ( ( uint8_t * ) & fsp - > file_id ,
sizeof ( fsp - > file_id ) ) ,
brl_get_locks_readonly_parser , & state ) ;
2014-02-26 16:30:25 -08:00
if ( NT_STATUS_EQUAL ( status , NT_STATUS_NOT_FOUND ) ) {
/*
* No locks on this file . Return an empty br_lock .
*/
br_lock = talloc ( fsp , struct byte_range_lock ) ;
if ( br_lock = = NULL ) {
goto fail ;
}
br_lock - > have_read_oplocks = false ;
br_lock - > num_locks = 0 ;
br_lock - > lock_data = NULL ;
} else if ( ! NT_STATUS_IS_OK ( status ) ) {
2013-09-11 11:36:54 +00:00
DEBUG ( 3 , ( " Could not parse byte range lock record: "
" %s \n " , nt_errstr ( status ) ) ) ;
goto fail ;
}
if ( br_lock = = NULL ) {
goto fail ;
}
2009-11-16 09:40:47 +01:00
}
2013-09-11 11:36:54 +00:00
br_lock - > fsp = fsp ;
br_lock - > modified = false ;
br_lock - > record = NULL ;
if ( lp_clustering ( ) ) {
/*
* In the cluster case we can ' t cache the brlock struct
* because dbwrap_get_seqnum does not work reliably over
* ctdb . Thus we have to throw away the brlock struct soon .
*/
talloc_steal ( talloc_tos ( ) , br_lock ) ;
} else {
/*
* Cache the brlock struct , invalidated when the dbwrap_seqnum
* changes . See beginning of this routine .
*/
TALLOC_FREE ( fsp - > brlock_rec ) ;
fsp - > brlock_rec = br_lock ;
fsp - > brlock_seqnum = dbwrap_get_seqnum ( brlock_db ) ;
}
2009-11-16 09:40:47 +01:00
2013-09-11 11:36:54 +00:00
fail :
TALLOC_FREE ( rw ) ;
return br_lock ;
2006-07-31 20:58:02 +00:00
}
2007-05-29 14:49:19 +00:00
struct brl_revalidate_state {
ssize_t array_size ;
uint32 num_pids ;
struct server_id * pids ;
} ;
/*
* Collect PIDs of all processes with pending entries
*/
static void brl_revalidate_collect ( struct file_id id , struct server_id pid ,
enum brl_type lock_type ,
enum brl_flavour lock_flav ,
br_off start , br_off size ,
void * private_data )
{
struct brl_revalidate_state * state =
( struct brl_revalidate_state * ) private_data ;
if ( ! IS_PENDING_LOCK ( lock_type ) ) {
return ;
}
add_to_large_array ( state , sizeof ( pid ) , ( void * ) & pid ,
& state - > pids , & state - > num_pids ,
& state - > array_size ) ;
}
/*
* qsort callback to sort the processes
*/
static int compare_procids ( const void * p1 , const void * p2 )
{
2011-05-05 14:22:11 -07:00
const struct server_id * i1 = ( const struct server_id * ) p1 ;
const struct server_id * i2 = ( const struct server_id * ) p2 ;
2007-05-29 14:49:19 +00:00
if ( i1 - > pid < i2 - > pid ) return - 1 ;
2013-11-09 21:02:10 +01:00
if ( i1 - > pid > i2 - > pid ) return 1 ;
2007-05-29 14:49:19 +00:00
return 0 ;
}
/*
* Send a MSG_SMB_UNLOCK message to all processes with pending byte range
* locks so that they retry . Mainly used in the cluster code after a node has
* died .
*
* Done in two steps to avoid double - sends : First we collect all entries in an
* array , then qsort that array and only send to non - dupes .
*/
2011-12-13 09:24:31 +01:00
void brl_revalidate ( struct messaging_context * msg_ctx ,
void * private_data ,
uint32_t msg_type ,
struct server_id server_id ,
DATA_BLOB * data )
2007-05-29 14:49:19 +00:00
{
struct brl_revalidate_state * state ;
uint32 i ;
struct server_id last_pid ;
2011-06-07 11:44:43 +10:00
if ( ! ( state = talloc_zero ( NULL , struct brl_revalidate_state ) ) ) {
2007-05-29 14:49:19 +00:00
DEBUG ( 0 , ( " talloc failed \n " ) ) ;
return ;
}
brl_forall ( brl_revalidate_collect , state ) ;
if ( state - > array_size = = - 1 ) {
DEBUG ( 0 , ( " talloc failed \n " ) ) ;
goto done ;
}
if ( state - > num_pids = = 0 ) {
goto done ;
}
2010-02-14 10:02:35 +11:00
TYPESAFE_QSORT ( state - > pids , state - > num_pids , compare_procids ) ;
2007-05-29 14:49:19 +00:00
ZERO_STRUCT ( last_pid ) ;
for ( i = 0 ; i < state - > num_pids ; i + + ) {
2012-06-16 00:26:26 +02:00
if ( serverid_equal ( & last_pid , & state - > pids [ i ] ) ) {
2007-05-29 14:49:19 +00:00
/*
* We ' ve seen that one already
*/
continue ;
}
messaging_send ( msg_ctx , state - > pids [ i ] , MSG_SMB_UNLOCK ,
& data_blob_null ) ;
last_pid = state - > pids [ i ] ;
}
done :
TALLOC_FREE ( state ) ;
return ;
}
2013-03-13 14:47:18 +01:00
bool brl_cleanup_disconnected ( struct file_id fid , uint64_t open_persistent_id )
{
bool ret = false ;
TALLOC_CTX * frame = talloc_stackframe ( ) ;
TDB_DATA key , val ;
struct db_record * rec ;
struct lock_struct * lock ;
unsigned n , num ;
NTSTATUS status ;
key = make_tdb_data ( ( void * ) & fid , sizeof ( fid ) ) ;
rec = dbwrap_fetch_locked ( brlock_db , frame , key ) ;
if ( rec = = NULL ) {
DEBUG ( 5 , ( " brl_cleanup_disconnected: failed to fetch record "
" for file %s \n " , file_id_string ( frame , & fid ) ) ) ;
goto done ;
}
val = dbwrap_record_get_value ( rec ) ;
lock = ( struct lock_struct * ) val . dptr ;
num = val . dsize / sizeof ( struct lock_struct ) ;
if ( lock = = NULL ) {
DEBUG ( 10 , ( " brl_cleanup_disconnected: no byte range locks for "
" file %s \n " , file_id_string ( frame , & fid ) ) ) ;
ret = true ;
goto done ;
}
for ( n = 0 ; n < num ; n + + ) {
struct lock_context * ctx = & lock [ n ] . context ;
if ( ! server_id_is_disconnected ( & ctx - > pid ) ) {
DEBUG ( 5 , ( " brl_cleanup_disconnected: byte range lock "
" %s used by server %s, do not cleanup \n " ,
file_id_string ( frame , & fid ) ,
server_id_str ( frame , & ctx - > pid ) ) ) ;
goto done ;
}
if ( ctx - > smblctx ! = open_persistent_id ) {
DEBUG ( 5 , ( " brl_cleanup_disconnected: byte range lock "
" %s expected smblctx %llu but found %llu "
" , do not cleanup \n " ,
file_id_string ( frame , & fid ) ,
( unsigned long long ) open_persistent_id ,
( unsigned long long ) ctx - > smblctx ) ) ;
goto done ;
}
}
status = dbwrap_record_delete ( rec ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 5 , ( " brl_cleanup_disconnected: failed to delete record "
" for file %s from %s, open %llu: %s \n " ,
file_id_string ( frame , & fid ) , dbwrap_name ( brlock_db ) ,
( unsigned long long ) open_persistent_id ,
nt_errstr ( status ) ) ) ;
goto done ;
}
DEBUG ( 10 , ( " brl_cleanup_disconnected: "
" file %s cleaned up %u entries from open %llu \n " ,
file_id_string ( frame , & fid ) , num ,
( unsigned long long ) open_persistent_id ) ) ;
ret = true ;
done :
talloc_free ( frame ) ;
return ret ;
}