nginx-quic/src/core/ngx_garbage_collector.c

218 lines
6.2 KiB
C
Raw Normal View History

2003-11-14 13:52:04 -03:00
/*
* Copyright (C) Igor Sysoev
*/
2003-11-14 13:52:04 -03:00
#include <ngx_config.h>
#include <ngx_core.h>
ngx_int_t ngx_collect_garbage(ngx_gc_t *ctx, ngx_str_t *dname, ngx_int_t level)
2003-11-14 13:52:04 -03:00
{
2003-11-21 03:30:49 -03:00
int rc;
2004-03-16 03:10:12 -04:00
u_char *last;
2003-11-21 03:30:49 -03:00
size_t len;
2003-11-16 18:49:42 -03:00
ngx_err_t err;
ngx_str_t fname, buf;
ngx_dir_t dir;
2003-11-14 13:52:04 -03:00
2003-11-16 18:49:42 -03:00
buf.len = 0;
2004-03-16 03:10:12 -04:00
#if (NGX_SUPPRESS_WARN)
buf.data = NULL;
fname.data = NULL;
#endif
2003-11-14 13:52:04 -03:00
2004-02-11 14:08:49 -03:00
ngx_log_debug2(NGX_LOG_DEBUG_CORE, ctx->log, 0,
"gc dir \"%s\":%d", dname->data, dname->len);
2003-11-14 13:52:04 -03:00
2003-11-16 18:49:42 -03:00
if (ngx_open_dir(dname, &dir) == NGX_ERROR) {
ngx_log_error(NGX_LOG_CRIT, ctx->log, ngx_errno,
ngx_open_dir_n " \"%s\" failed", dname->data);
2003-11-14 13:52:04 -03:00
return NGX_ERROR;
}
for ( ;; ) {
2003-11-16 18:49:42 -03:00
ngx_set_errno(0);
if (ngx_read_dir(&dir) == NGX_ERROR) {
err = ngx_errno;
if (err != NGX_ENOMOREFILES) {
ngx_log_error(NGX_LOG_CRIT, ctx->log, err,
ngx_read_dir_n " \"%s\" failed", dname->data);
rc = NGX_ERROR;
2003-11-14 13:52:04 -03:00
2003-11-16 18:49:42 -03:00
} else {
rc = NGX_OK;
2003-11-14 13:52:04 -03:00
}
2003-11-16 18:49:42 -03:00
2003-11-14 13:52:04 -03:00
break;
}
2003-11-16 18:49:42 -03:00
len = ngx_de_namelen(&dir);
2003-11-14 13:52:04 -03:00
2004-02-11 14:08:49 -03:00
ngx_log_debug2(NGX_LOG_DEBUG_CORE, ctx->log, 0,
"gc name \"%s\":%d", ngx_de_name(&dir), len);
2003-11-14 13:52:04 -03:00
2003-11-16 18:49:42 -03:00
if (len == 1 && ngx_de_name(&dir)[0] == '.') {
2003-11-14 13:52:04 -03:00
continue;
}
2003-11-16 18:49:42 -03:00
if (len == 2
&& ngx_de_name(&dir)[0] == '.'
&& ngx_de_name(&dir)[1] == '.')
{
2003-11-14 13:52:04 -03:00
continue;
}
2003-11-16 18:49:42 -03:00
fname.len = dname->len + 1+ len;
if (fname.len + NGX_DIR_MASK_LEN > buf.len) {
if (buf.len) {
ngx_free(buf.data);
2003-11-14 13:52:04 -03:00
}
2003-11-16 18:49:42 -03:00
buf.len = dname->len + 1 + len + NGX_DIR_MASK_LEN;
2003-11-14 13:52:04 -03:00
buf.data = ngx_alloc(buf.len + 1, ctx->log);
if (buf.data == NULL) {
2003-11-14 13:52:04 -03:00
return NGX_ABORT;
}
}
2003-11-16 18:49:42 -03:00
last = ngx_cpymem(buf.data, dname->data, dname->len);
2003-11-14 13:52:04 -03:00
*last++ = '/';
2003-11-16 18:49:42 -03:00
ngx_memcpy(last, ngx_de_name(&dir), len + 1);
fname.data = buf.data;
2003-11-14 13:52:04 -03:00
2004-02-11 14:08:49 -03:00
ngx_log_debug1(NGX_LOG_DEBUG_CORE, ctx->log, 0,
"gc path: \"%s\"", fname.data);
2003-11-14 13:52:04 -03:00
if (!dir.valid_info) {
2003-11-16 18:49:42 -03:00
if (ngx_de_info(fname.data, &dir) == NGX_FILE_ERROR) {
ngx_log_error(NGX_LOG_CRIT, ctx->log, ngx_errno,
ngx_de_info_n " \"%s\" failed", fname.data);
continue;
}
2003-11-14 13:52:04 -03:00
}
2003-11-16 18:49:42 -03:00
if (ngx_de_is_dir(&dir)) {
2003-11-14 13:52:04 -03:00
2004-02-11 14:08:49 -03:00
ngx_log_debug1(NGX_LOG_DEBUG_CORE, ctx->log, 0,
"gc enter dir \"%s\"", fname.data);
2003-11-14 13:52:04 -03:00
if (level == -1
/* there can not be directory on the last level */
|| level == NGX_MAX_PATH_LEVEL
/* an directory from the old path hierarchy */
2003-11-16 18:49:42 -03:00
|| len != ctx->path->level[level])
2003-11-14 13:52:04 -03:00
{
if (ngx_collect_garbage(ctx, &fname, -1) == NGX_ABORT) {
return NGX_ABORT;
}
2003-11-16 18:49:42 -03:00
fname.data[fname.len] = '\0';
2003-11-14 13:52:04 -03:00
ngx_log_error(NGX_LOG_NOTICE, ctx->log, 0,
"delete old hierachy directory \"%s\"",
fname.data);
if (ngx_delete_dir(fname.data) == NGX_FILE_ERROR) {
ngx_log_error(NGX_LOG_CRIT, ctx->log, ngx_errno,
ngx_delete_dir_n " \"%s\" failed",
fname.data);
} else {
ctx->deleted++;
2003-11-16 18:49:42 -03:00
ctx->freed += ngx_de_size(&dir);
2003-11-14 13:52:04 -03:00
}
continue;
}
if (ngx_collect_garbage(ctx, &fname, level + 1) == NGX_ABORT) {
return NGX_ABORT;
}
2003-11-16 18:49:42 -03:00
} else if (ngx_de_is_file(&dir)) {
2004-02-11 14:08:49 -03:00
ngx_log_debug1(NGX_LOG_DEBUG_CORE, ctx->log, 0,
"gc file \"%s\"", fname.data);
2003-11-14 13:52:04 -03:00
if (level == -1
|| (level < NGX_MAX_PATH_LEVEL && ctx->path->level[level] != 0))
{
if (ngx_delete_file(fname.data) == NGX_FILE_ERROR) {
ngx_log_error(NGX_LOG_CRIT, ctx->log, ngx_errno,
ngx_delete_file_n " \"%s\" failed",
fname.data);
} else {
ctx->deleted++;
2003-11-16 18:49:42 -03:00
ctx->freed += ngx_de_size(&dir);
2003-11-14 13:52:04 -03:00
}
continue;
}
2003-11-16 18:49:42 -03:00
if (ctx->handler(ctx, &fname, &dir) == NGX_ABORT) {
2003-11-14 13:52:04 -03:00
return NGX_ABORT;
}
} else {
ngx_log_error(NGX_LOG_CRIT, ctx->log, ngx_errno,
"the file \"%s\" has unknown type, deleting",
fname.data);
2003-11-14 13:52:04 -03:00
if (ngx_delete_file(fname.data) == NGX_FILE_ERROR) {
ngx_log_error(NGX_LOG_CRIT, ctx->log, ngx_errno,
ngx_delete_file_n " \"%s\" failed", fname.data);
} else {
ctx->deleted++;
2003-11-16 18:49:42 -03:00
ctx->freed += ngx_de_size(&dir);
2003-11-14 13:52:04 -03:00
}
}
}
2003-11-16 18:49:42 -03:00
if (buf.len) {
ngx_free(buf.data);
}
if (ngx_close_dir(&dir) == NGX_ERROR) {
ngx_log_error(NGX_LOG_CRIT, ctx->log, ngx_errno,
ngx_close_dir_n " \"%s\" failed", fname.data);
}
return rc;
2003-11-14 13:52:04 -03:00
}
ngx_int_t ngx_garbage_collector_temp_handler(ngx_gc_t *ctx, ngx_str_t *name,
ngx_dir_t *dir)
2003-11-14 13:52:04 -03:00
{
/*
2004-02-11 14:08:49 -03:00
* We use mtime only and do not use atime because:
2003-11-14 13:52:04 -03:00
* on NTFS access time has a resolution of 1 hour,
* on NT FAT access time has a resolution of 1 day,
2004-02-11 14:08:49 -03:00
* Unices have the mount option "noatime".
2003-11-14 13:52:04 -03:00
*/
2004-06-27 14:01:57 -04:00
if (ngx_time() - ngx_de_mtime(dir) < 3600) {
2003-11-14 13:52:04 -03:00
return NGX_OK;
}
ngx_log_error(NGX_LOG_NOTICE, ctx->log, 0,
"delete the stale temporary file \"%s\"", name->data);
2003-11-14 13:52:04 -03:00
if (ngx_delete_file(name->data) == NGX_FILE_ERROR) {
ngx_log_error(NGX_LOG_CRIT, ctx->log, ngx_errno,
ngx_delete_file_n " \"%s\" failed", name->data);
return NGX_ERROR;
}
ctx->deleted++;
2003-11-16 18:49:42 -03:00
ctx->freed += ngx_de_size(dir);
2003-11-18 18:34:08 -03:00
2003-11-14 13:52:04 -03:00
return NGX_OK;
}