27
27
TL_READ # Low priority read
28
28
TL_READ_WITH_SHARED_LOCKS
29
TL_READ_HIGH_PRIORITY # High priority read
29
30
TL_READ_NO_INSERT # Read without concurrent inserts
30
31
TL_WRITE_ALLOW_WRITE # Write lock that allows other writers
31
32
TL_WRITE_ALLOW_READ # Write lock, but allow reading
32
33
TL_WRITE_CONCURRENT_INSERT
33
34
# Insert that can be mixed when selects
35
TL_WRITE_DELAYED # Used by delayed insert
36
# Allows lower locks to take over
37
TL_WRITE_LOW_PRIORITY # Low priority write
34
38
TL_WRITE # High priority write
35
39
TL_WRITE_ONLY # High priority write
36
40
# Abort all new lock request with an error
135
131
lock->write_wait.last= &lock->write_wait.data;
136
132
lock->write.last= &lock->write.data;
138
pthread_mutex_lock(&internal::THR_LOCK_lock); /* Add to locks in use */
139
thr_lock_thread_list.push_front(lock);
140
pthread_mutex_unlock(&internal::THR_LOCK_lock);
134
pthread_mutex_lock(&THR_LOCK_lock); /* Add to locks in use */
135
lock->list.data=(void*) lock;
136
thr_lock_thread_list=list_add(thr_lock_thread_list,&lock->list);
137
pthread_mutex_unlock(&THR_LOCK_lock);
144
142
void thr_lock_delete(THR_LOCK *lock)
146
144
pthread_mutex_destroy(&lock->mutex);
147
pthread_mutex_lock(&internal::THR_LOCK_lock);
148
thr_lock_thread_list.remove(lock);
149
pthread_mutex_unlock(&internal::THR_LOCK_lock);
145
pthread_mutex_lock(&THR_LOCK_lock);
146
thr_lock_thread_list=list_delete(thr_lock_thread_list,&lock->list);
147
pthread_mutex_unlock(&THR_LOCK_lock);
153
152
void thr_lock_info_init(THR_LOCK_INFO *info)
155
internal::st_my_thread_var *tmp= my_thread_var;
156
info->thread= tmp->pthread_self;
154
struct st_my_thread_var *tmp= my_thread_var;
155
info->thread= tmp->pthread_self;
157
156
info->thread_id= tmp->id;
158
157
info->n_cursors= 0;
262
275
/* The following must be done after unlock of lock->mutex */
263
276
pthread_mutex_lock(&thread_var->mutex);
264
thread_var->current_mutex= NULL;
265
thread_var->current_cond= NULL;
277
thread_var->current_mutex= 0;
278
thread_var->current_cond= 0;
266
279
pthread_mutex_unlock(&thread_var->mutex);
271
static enum enum_thr_lock_result
284
enum enum_thr_lock_result
272
285
thr_lock(THR_LOCK_DATA *data, THR_LOCK_OWNER *owner,
273
286
enum thr_lock_type lock_type)
344
358
else /* Request for WRITE lock */
346
if (lock_type == TL_WRITE_CONCURRENT_INSERT && ! lock->check_status)
360
if (lock_type == TL_WRITE_DELAYED)
362
if (lock->write.data && lock->write.data->type == TL_WRITE_ONLY)
364
data->type=TL_UNLOCK;
365
result= THR_LOCK_ABORTED; /* Can't wait for this one */
369
if there is a TL_WRITE_ALLOW_READ lock, we have to wait for a lock
370
(TL_WRITE_ALLOW_READ is used for ALTER TABLE in MySQL)
372
if ((!lock->write.data ||
373
lock->write.data->type != TL_WRITE_ALLOW_READ) &&
374
!have_specific_lock(lock->write_wait.data,TL_WRITE_ALLOW_READ) &&
375
(lock->write.data || lock->read.data))
377
/* Add delayed write lock to write_wait queue, and return at once */
378
(*lock->write_wait.last)=data;
379
data->prev=lock->write_wait.last;
380
lock->write_wait.last= &data->next;
381
data->cond=get_cond();
383
We don't have to do get_status here as we will do it when we change
384
the delayed lock to a real write lock
386
statistic_increment(locks_immediate,&THR_LOCK_lock);
390
else if (lock_type == TL_WRITE_CONCURRENT_INSERT && ! lock->check_status)
347
391
data->type=lock_type= thr_upgraded_concurrent_insert_lock;
349
393
if (lock->write.data) /* If there is a write lock */
739
793
TL_WRITE_ONLY to abort any new accesses to the lock
742
void thr_abort_locks(THR_LOCK *lock)
796
void thr_abort_locks(THR_LOCK *lock, bool upgrade_lock)
744
798
THR_LOCK_DATA *data;
745
799
pthread_mutex_lock(&lock->mutex);
747
801
for (data=lock->read_wait.data; data ; data=data->next)
749
data->type= TL_UNLOCK; /* Mark killed */
803
data->type=TL_UNLOCK; /* Mark killed */
750
804
/* It's safe to signal the cond first: we're still holding the mutex. */
751
805
pthread_cond_signal(data->cond);
752
data->cond= NULL; /* Removed from list */
806
data->cond=0; /* Removed from list */
754
808
for (data=lock->write_wait.data; data ; data=data->next)
756
810
data->type=TL_UNLOCK;
757
811
pthread_cond_signal(data->cond);
760
814
lock->read_wait.last= &lock->read_wait.data;
761
815
lock->write_wait.last= &lock->write_wait.data;
762
816
lock->read_wait.data=lock->write_wait.data=0;
763
if (lock->write.data)
817
if (upgrade_lock && lock->write.data)
764
818
lock->write.data->type=TL_WRITE_ONLY;
765
819
pthread_mutex_unlock(&lock->mutex);
818
} /* namespace drizzled */
874
Downgrade a WRITE_* to a lower WRITE level
876
thr_downgrade_write_lock()
877
in_data Lock data of thread downgrading its lock
878
new_lock_type New write lock type
882
This can be used to downgrade a lock already owned. When the downgrade
883
occurs also other waiters, both readers and writers can be allowed to
885
The previous lock is often TL_WRITE_ONLY but can also be
886
TL_WRITE and TL_WRITE_ALLOW_READ. The normal downgrade variants are
887
TL_WRITE_ONLY => TL_WRITE_ALLOW_READ After a short exclusive lock
888
TL_WRITE_ALLOW_READ => TL_WRITE_ALLOW_WRITE After discovering that the
889
operation didn't need such a high lock.
890
TL_WRITE_ONLY => TL_WRITE after a short exclusive lock while holding a
892
TL_WRITE_ONLY => TL_WRITE_ALLOW_WRITE After a short exclusive lock after
893
already earlier having dongraded lock to TL_WRITE_ALLOW_WRITE
894
The implementation is conservative and rather don't start rather than
895
go on unknown paths to start, the common cases are handled.
898
In its current implementation it is only allowed to downgrade from
899
TL_WRITE_ONLY. In this case there are no waiters. Thus no wake up
903
void thr_downgrade_write_lock(THR_LOCK_DATA *in_data,
904
enum thr_lock_type new_lock_type)
906
THR_LOCK *lock=in_data->lock;
908
pthread_mutex_lock(&lock->mutex);
909
in_data->type= new_lock_type;
911
pthread_mutex_unlock(&lock->mutex);
915
/* Upgrade a WRITE_DELAY lock to a WRITE_LOCK */
917
bool thr_upgrade_write_delay_lock(THR_LOCK_DATA *data)
919
THR_LOCK *lock=data->lock;
921
pthread_mutex_lock(&lock->mutex);
922
if (data->type == TL_UNLOCK || data->type >= TL_WRITE_LOW_PRIORITY)
924
pthread_mutex_unlock(&lock->mutex);
925
return(data->type == TL_UNLOCK); /* Test if Aborted */
927
/* TODO: Upgrade to TL_WRITE_CONCURRENT_INSERT in some cases */
928
data->type=TL_WRITE; /* Upgrade lock */
930
/* Check if someone has given us the lock */
933
if (!lock->read.data) /* No read locks */
934
{ /* We have the lock */
935
if (data->lock->get_status)
936
(*data->lock->get_status)(data->status_param, 0);
937
pthread_mutex_unlock(&lock->mutex);
941
if (((*data->prev)=data->next)) /* remove from lock-list */
942
data->next->prev= data->prev;
944
lock->write.last=data->prev;
946
if ((data->next=lock->write_wait.data)) /* Put first in lock_list */
947
data->next->prev= &data->next;
949
lock->write_wait.last= &data->next;
950
data->prev= &lock->write_wait.data;
951
lock->write_wait.data=data;
954
return(wait_for_lock(&lock->write_wait,data,1));
958
/* downgrade a WRITE lock to a WRITE_DELAY lock if there is pending locks */
960
bool thr_reschedule_write_lock(THR_LOCK_DATA *data)
962
THR_LOCK *lock=data->lock;
964
pthread_mutex_lock(&lock->mutex);
965
if (!lock->read_wait.data) /* No waiting read locks */
967
pthread_mutex_unlock(&lock->mutex);
971
data->type=TL_WRITE_DELAYED;
972
if (lock->update_status)
973
(*lock->update_status)(data->status_param);
974
if (((*data->prev)=data->next)) /* remove from lock-list */
975
data->next->prev= data->prev;
977
lock->write.last=data->prev;
979
if ((data->next=lock->write_wait.data)) /* Put first in lock_list */
980
data->next->prev= &data->next;
982
lock->write_wait.last= &data->next;
983
data->prev= &lock->write_wait.data;
984
data->cond=get_cond(); /* This was zero */
985
lock->write_wait.data=data;
986
free_all_read_locks(lock,0);
988
pthread_mutex_unlock(&lock->mutex);
989
return(thr_upgrade_write_delay_lock(data));