HelenOS sources
This source file includes following definitions.
- fmgt_create
- fmgt_set_cb
- fmgt_set_init_update
- fmgt_destroy
- fmgt_new_file_suggest
- fmgt_get_progress
- fmgt_progress_update
- fmgt_initial_progress_update
- fmgt_final_progress_update
- fmgt_timer_fun
- fmgt_timer_start
- fmgt_new_file
#include <dirent.h>
#include <errno.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <stddef.h>
#include <str.h>
#include <vfs/vfs.h>
#include <dirent.h>
#include "fmgt.h"
#define NEWNAME_LEN 64
#define BUFFER_SIZE 16384
errno_t fmgt_create(fmgt_t **rfmgt)
{
fmgt_t *fmgt;
fmgt = calloc(1, sizeof(fmgt_t));
if (fmgt == NULL)
return ENOMEM;
fibril_mutex_initialize(&fmgt->lock);
fmgt->timer = fibril_timer_create(&fmgt->lock);
if (fmgt->timer == NULL) {
free(fmgt);
return ENOMEM;
}
*rfmgt = fmgt;
return EOK;
}
void fmgt_set_cb(fmgt_t *fmgt, fmgt_cb_t *cb, void *arg)
{
fmgt->cb = cb;
fmgt->cb_arg = arg;
}
void fmgt_set_init_update(fmgt_t *fmgt, bool enabled)
{
fmgt->do_init_update = enabled;
}
void fmgt_destroy(fmgt_t *fmgt)
{
(void)fibril_timer_clear(fmgt->timer);
fibril_timer_destroy(fmgt->timer);
free(fmgt);
}
errno_t fmgt_new_file_suggest(char **rstr)
{
errno_t rc;
vfs_stat_t stat;
unsigned u;
char name[NEWNAME_LEN];
u = 0;
while (true) {
snprintf(name, sizeof(name), "noname%02u.txt", u);
rc = vfs_stat_path(name, &stat);
if (rc != EOK)
break;
++u;
}
*rstr = str_dup(name);
if (*rstr == NULL)
return ENOMEM;
return EOK;
}
static void fmgt_get_progress(fmgt_t *fmgt, fmgt_progress_t *progress)
{
unsigned percent;
if (fmgt->curf_totalb > 0)
percent = fmgt->curf_procb * 100 / fmgt->curf_totalb;
else
percent = 100;
capa_blocks_format_buf(fmgt->curf_procb, 1, progress->curf_procb,
sizeof(progress->curf_procb));
capa_blocks_format_buf(fmgt->curf_totalb, 1, progress->curf_totalb,
sizeof(progress->curf_totalb));
snprintf(progress->curf_percent, sizeof(progress->curf_percent), "%u%%",
percent);
}
static void fmgt_progress_update(fmgt_t *fmgt)
{
fmgt_progress_t progress;
if (fmgt->cb != NULL && fmgt->cb->progress != NULL) {
fmgt_get_progress(fmgt, &progress);
fmgt->curf_progr = true;
fmgt->cb->progress(fmgt->cb_arg, &progress);
}
}
static void fmgt_initial_progress_update(fmgt_t *fmgt)
{
if (fmgt->do_init_update)
fmgt_progress_update(fmgt);
}
static void fmgt_final_progress_update(fmgt_t *fmgt)
{
if (fmgt->curf_progr)
fmgt_progress_update(fmgt);
}
static void fmgt_timer_fun(void *arg)
{
fmgt_t *fmgt = (fmgt_t *)arg;
fmgt_progress_update(fmgt);
fibril_timer_set(fmgt->timer, 500000, fmgt_timer_fun, (void *)fmgt);
}
static void fmgt_timer_start(fmgt_t *fmgt)
{
fibril_timer_set(fmgt->timer, 500000, fmgt_timer_fun, (void *)fmgt);
}
errno_t fmgt_new_file(fmgt_t *fmgt, const char *fname, uint64_t fsize)
{
int fd;
size_t nw;
aoff64_t pos = 0;
uint64_t now;
char *buffer;
errno_t rc;
buffer = calloc(BUFFER_SIZE, 1);
if (buffer == NULL)
return ENOMEM;
rc = vfs_lookup_open(fname, WALK_REGULAR | WALK_MUST_CREATE,
MODE_WRITE, &fd);
if (rc != EOK) {
free(buffer);
return rc;
}
fmgt->curf_procb = 0;
fmgt->curf_totalb = fsize;
fmgt->curf_progr = false;
fmgt_timer_start(fmgt);
fmgt_initial_progress_update(fmgt);
while (fmgt->curf_procb < fsize) {
now = fsize - fmgt->curf_procb;
if (now > BUFFER_SIZE)
now = BUFFER_SIZE;
rc = vfs_write(fd, &pos, buffer, now, &nw);
if (rc != EOK) {
free(buffer);
vfs_put(fd);
fmgt_final_progress_update(fmgt);
return rc;
}
fmgt->curf_procb += nw;
}
free(buffer);
vfs_put(fd);
fmgt_final_progress_update(fmgt);
return EOK;
}
HelenOS homepage, sources at GitHub