forgot to add these...
This commit is contained in:
parent
59b133dfd1
commit
f06159b3cd
2 changed files with 232 additions and 0 deletions
77
lladd/truncation.h
Normal file
77
lladd/truncation.h
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
/*---
|
||||||
|
This software is copyrighted by the Regents of the University of
|
||||||
|
California, and other parties. The following terms apply to all files
|
||||||
|
associated with the software unless explicitly disclaimed in
|
||||||
|
individual files.
|
||||||
|
|
||||||
|
The authors hereby grant permission to use, copy, modify, distribute,
|
||||||
|
and license this software and its documentation for any purpose,
|
||||||
|
provided that existing copyright notices are retained in all copies
|
||||||
|
and that this notice is included verbatim in any distributions. No
|
||||||
|
written agreement, license, or royalty fee is required for any of the
|
||||||
|
authorized uses. Modifications to this software may be copyrighted by
|
||||||
|
their authors and need not follow the licensing terms described here,
|
||||||
|
provided that the new terms are clearly indicated on the first page of
|
||||||
|
each file where they apply.
|
||||||
|
|
||||||
|
IN NO EVENT SHALL THE AUTHORS OR DISTRIBUTORS BE LIABLE TO ANY PARTY
|
||||||
|
FOR DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
|
||||||
|
ARISING OUT OF THE USE OF THIS SOFTWARE, ITS DOCUMENTATION, OR ANY
|
||||||
|
DERIVATIVES THEREOF, EVEN IF THE AUTHORS HAVE BEEN ADVISED OF THE
|
||||||
|
POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
THE AUTHORS AND DISTRIBUTORS SPECIFICALLY DISCLAIM ANY WARRANTIES,
|
||||||
|
INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, AND
|
||||||
|
NON-INFRINGEMENT. THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, AND
|
||||||
|
THE AUTHORS AND DISTRIBUTORS HAVE NO OBLIGATION TO PROVIDE
|
||||||
|
MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
|
||||||
|
|
||||||
|
GOVERNMENT USE: If you are acquiring this software on behalf of the
|
||||||
|
U.S. government, the Government shall have only "Restricted Rights" in
|
||||||
|
the software and related documentation as defined in the Federal
|
||||||
|
Acquisition Regulations (FARs) in Clause 52.227.19 (c) (2). If you are
|
||||||
|
acquiring the software on behalf of the Department of Defense, the
|
||||||
|
software shall be classified as "Commercial Computer Software" and the
|
||||||
|
Government shall have only "Restricted Rights" as defined in Clause
|
||||||
|
252.227-7013 (c) (1) of DFARs. Notwithstanding the foregoing, the
|
||||||
|
authors grant the U.S. Government and others acting in its behalf
|
||||||
|
permission to use and distribute the software in accordance with the
|
||||||
|
terms specified in this license.
|
||||||
|
---*/
|
||||||
|
/**
|
||||||
|
* @file
|
||||||
|
*
|
||||||
|
* Implementation of log truncation for lladd.
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <lladd/transactional.h>
|
||||||
|
|
||||||
|
#ifndef __LLADD_TRUNCATION_H
|
||||||
|
#define __LLADD_TRUNCATION_H 1
|
||||||
|
|
||||||
|
void dirtyPagesInit();
|
||||||
|
void dirtyPagesDeinit();
|
||||||
|
|
||||||
|
void dirtyPages_add(Page * p);
|
||||||
|
void dirtyPages_remove(Page * p);
|
||||||
|
|
||||||
|
void truncationInit();
|
||||||
|
void truncationDeinit();
|
||||||
|
|
||||||
|
extern int lladd_enableAutoTruncation;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Spawn a periodic, demand-based log truncation thread.
|
||||||
|
*/
|
||||||
|
void autoTruncate();
|
||||||
|
/**
|
||||||
|
Initiate a round of log truncation.
|
||||||
|
*/
|
||||||
|
int truncateNow();
|
||||||
|
|
||||||
|
|
||||||
|
#endif
|
155
src/lladd/truncation.c
Normal file
155
src/lladd/truncation.c
Normal file
|
@ -0,0 +1,155 @@
|
||||||
|
#include <lladd/truncation.h>
|
||||||
|
#include <pbl/pbl.h>
|
||||||
|
#include <lladd/logger/logger2.h>
|
||||||
|
#include "page.h"
|
||||||
|
#include <assert.h>
|
||||||
|
#include "pageFile.h"
|
||||||
|
|
||||||
|
volatile static int initialized = 0;
|
||||||
|
static int automaticallyTuncating = 0;
|
||||||
|
static pthread_t truncationThread;
|
||||||
|
|
||||||
|
static pblHashTable_t * dirtyPages = 0;
|
||||||
|
static pthread_mutex_t dirtyPages_mutex = PTHREAD_MUTEX_INITIALIZER;
|
||||||
|
|
||||||
|
int lladd_enableAutoTruncation = 1;
|
||||||
|
#define TARGET_LOG_SIZE (1024 * 1024 * 50)
|
||||||
|
#define TRUNCATE_INTERVAL 1
|
||||||
|
#define MIN_INCREMENTAL_TRUNCATION (1024 * 1024 * 10)
|
||||||
|
void dirtyPages_add(Page * p) {
|
||||||
|
pthread_mutex_lock(&dirtyPages_mutex);
|
||||||
|
if(!p->dirty) {
|
||||||
|
p->dirty = 1;
|
||||||
|
|
||||||
|
assert(!pblHtLookup(dirtyPages, &(p->id), sizeof(int)));
|
||||||
|
pblHtInsert(dirtyPages, &(p->id), sizeof(int), (void*)p->LSN);
|
||||||
|
}
|
||||||
|
pthread_mutex_unlock(&dirtyPages_mutex);
|
||||||
|
}
|
||||||
|
|
||||||
|
void dirtyPages_remove(Page * p) {
|
||||||
|
pthread_mutex_lock(&dirtyPages_mutex);
|
||||||
|
// printf("Removing page %d\n", p->id);
|
||||||
|
//assert(pblHtLookup(dirtyPages, &(p->id), sizeof(int)));
|
||||||
|
// printf("With lsn = %d\n", (lsn_t)pblHtCurrent(dirtyPages));
|
||||||
|
assert(!pblHtRemove(dirtyPages, &(p->id), sizeof(int)));
|
||||||
|
//assert(!pblHtLookup(dirtyPages, &(p->id), sizeof(int)));
|
||||||
|
pthread_mutex_unlock(&dirtyPages_mutex);
|
||||||
|
}
|
||||||
|
|
||||||
|
static lsn_t dirtyPages_minRecLSN() {
|
||||||
|
lsn_t lsn = LogFlushedLSN ();
|
||||||
|
int* pageid;
|
||||||
|
pthread_mutex_lock(&dirtyPages_mutex);
|
||||||
|
|
||||||
|
for( pageid = (int*)pblHtFirst (dirtyPages); pageid; pageid = (int*)pblHtNext(dirtyPages)) {
|
||||||
|
lsn_t thisLSN = (lsn_t) pblHtCurrent(dirtyPages);
|
||||||
|
// printf("lsn = %d\n", thisLSN);
|
||||||
|
if(thisLSN < lsn) {
|
||||||
|
lsn = thisLSN;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pthread_mutex_unlock(&dirtyPages_mutex);
|
||||||
|
|
||||||
|
return lsn;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void dirtyPages_flush() {
|
||||||
|
int * staleDirtyPages = malloc(sizeof(int) * MAX_BUFFER_SIZE);
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < MAX_BUFFER_SIZE+1; i++) {
|
||||||
|
staleDirtyPages[i] = -1;
|
||||||
|
}
|
||||||
|
Page* p = 0;
|
||||||
|
pthread_mutex_lock(&dirtyPages_mutex);
|
||||||
|
void* tmp;
|
||||||
|
i = 0;
|
||||||
|
|
||||||
|
for(tmp = pblHtFirst(dirtyPages); tmp; tmp = pblHtNext(dirtyPages)) {
|
||||||
|
staleDirtyPages[i] = *((int*) pblHtCurrentKey(dirtyPages));
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
assert(i <= MAX_BUFFER_SIZE);
|
||||||
|
pthread_mutex_unlock(&dirtyPages_mutex);
|
||||||
|
|
||||||
|
for(i = 0; i < MAX_BUFFER_SIZE && staleDirtyPages[i] != -1; i++) {
|
||||||
|
p = loadPage(-1, staleDirtyPages[i]);
|
||||||
|
//if(p->dirty) {
|
||||||
|
pageWrite(p);
|
||||||
|
// dirtyPages_remove(p);
|
||||||
|
//}
|
||||||
|
releasePage(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void dirtyPagesInit() {
|
||||||
|
dirtyPages = pblHtCreate();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void dirtyPagesDeinit() {
|
||||||
|
pblHtDelete(dirtyPages);
|
||||||
|
dirtyPages = 0;
|
||||||
|
}
|
||||||
|
void truncationInit() {
|
||||||
|
initialized = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void truncationDeinit() {
|
||||||
|
initialized = 0;
|
||||||
|
if(automaticallyTuncating) {
|
||||||
|
void * ret = 0;
|
||||||
|
pthread_join(truncationThread, &ret);
|
||||||
|
}
|
||||||
|
automaticallyTuncating = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void* periodicTruncation(void * ignored) {
|
||||||
|
while(initialized) {
|
||||||
|
if(LogFlushedLSN() - LogTruncationPoint() > TARGET_LOG_SIZE) {
|
||||||
|
truncateNow();
|
||||||
|
}
|
||||||
|
// @todo TRUNCATE_INTERVAL should be dynamically set...
|
||||||
|
sleep(TRUNCATE_INTERVAL);
|
||||||
|
}
|
||||||
|
return (void*)0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void autoTruncate() {
|
||||||
|
assert(!automaticallyTuncating);
|
||||||
|
automaticallyTuncating = 1;
|
||||||
|
pthread_create(&truncationThread, 0, &periodicTruncation, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int truncateNow() {
|
||||||
|
lsn_t page_rec_lsn = dirtyPages_minRecLSN();
|
||||||
|
lsn_t xact_rec_lsn = transactions_minRecLSN();
|
||||||
|
lsn_t rec_lsn = page_rec_lsn < xact_rec_lsn ? page_rec_lsn : xact_rec_lsn;
|
||||||
|
lsn_t log_trunc = LogTruncationPoint();
|
||||||
|
if((rec_lsn - log_trunc) > MIN_INCREMENTAL_TRUNCATION) {
|
||||||
|
printf("Truncating now. rec_lsn = %ld, log_trunc = %ld\n", rec_lsn, log_trunc);
|
||||||
|
LogTruncate(rec_lsn);
|
||||||
|
return 1;
|
||||||
|
} else {
|
||||||
|
lsn_t flushed = LogFlushedLSN();
|
||||||
|
if(flushed - log_trunc > 2 * TARGET_LOG_SIZE) {
|
||||||
|
printf("Flushing dirty buffers: rec_lsn = %ld log_trunc = %ld flushed = %ld\n", rec_lsn, log_trunc, flushed);
|
||||||
|
fflush(stdout);
|
||||||
|
dirtyPages_flush();
|
||||||
|
rec_lsn = dirtyPages_minRecLSN();
|
||||||
|
printf("Truncating to rec_lsn = %ld\n", rec_lsn);
|
||||||
|
fflush(stdout);
|
||||||
|
if(rec_lsn != flushed) {
|
||||||
|
LogTruncate(rec_lsn);
|
||||||
|
return 1;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue