mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-03 05:18:29 +03:00
216 lines
4.6 KiB
C
216 lines
4.6 KiB
C
/*
|
|
* dm-parse.c
|
|
*
|
|
* Copyright (C) 2001 Sistina Software
|
|
*
|
|
* This software is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2, or (at
|
|
* your option) any later version.
|
|
*
|
|
* This software 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.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with GNU CC; see the file COPYING. If not, write to
|
|
* the Free Software Foundation, 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*
|
|
* 4/09/2001 - First version [Joe Thornber]
|
|
*/
|
|
|
|
#include "dm.h"
|
|
|
|
struct dm_table *dm_parse(extract_line_fn line_fn, void *l_private)
|
|
{
|
|
struct text_region line, word;
|
|
struct dm_table *table = dm_table_create();
|
|
struct target_type *ttype;
|
|
offset_t start, size, high;
|
|
char target_name[64];
|
|
void *context;
|
|
int last_line_good = 1, was_error = 0;
|
|
|
|
if (table == NULL)
|
|
return NULL;
|
|
|
|
#define PARSE_ERROR {last_line_good = 0; was_error = 1; continue;}
|
|
|
|
while (line_fn(&line, l_private)) {
|
|
|
|
/*
|
|
* each line is of the format:
|
|
* <sector start> <length (sectors)> <target type> <args...>
|
|
*/
|
|
|
|
/* the line may be blank ... */
|
|
dm_eat_space(&line);
|
|
if (dm_empty_tok(&line) || (*line.b == '#'))
|
|
continue;
|
|
|
|
/* sector start */
|
|
if (!dm_get_number(&line, &start)) {
|
|
table->err_msg = "expecting a number for sector start";
|
|
PARSE_ERROR;
|
|
}
|
|
|
|
/* length */
|
|
if (!dm_get_number(&line, &size)) {
|
|
table->err_msg = "expecting a number for region length";
|
|
PARSE_ERROR;
|
|
}
|
|
|
|
/* target type */
|
|
if (!dm_get_word(&line, &word)) {
|
|
table->err_msg = "target type missing";
|
|
PARSE_ERROR;
|
|
}
|
|
|
|
/* we have to copy the target type to a C str */
|
|
dm_txt_copy(target_name, sizeof(target_name), &word);
|
|
|
|
/* lookup the target type */
|
|
if (!(ttype = dm_get_target_type(target_name))) {
|
|
table->err_msg = "unable to find target type";
|
|
PARSE_ERROR;
|
|
}
|
|
|
|
/* check there isn't a gap, but only if the last target
|
|
parsed ok. */
|
|
if (last_line_good &&
|
|
|
|
((table->num_targets &&
|
|
start != table->highs[table->num_targets - 1] + 1) ||
|
|
(!table->num_targets && start))) {
|
|
table->err_msg = "gap in target ranges";
|
|
PARSE_ERROR;
|
|
}
|
|
|
|
/* build the target */
|
|
context = ttype->ctr(table, start, size, &line);
|
|
if (IS_ERR(context)) {
|
|
PARSE_ERROR;
|
|
}
|
|
|
|
/* no point registering the target
|
|
if there was an error. */
|
|
if (was_error)
|
|
continue;
|
|
|
|
/* add the target to the table */
|
|
high = start + (size - 1);
|
|
if (dm_table_add_target(table, high, ttype, context)) {
|
|
table->err_msg = "internal error adding target to table";
|
|
PARSE_ERROR;
|
|
}
|
|
|
|
/* Ensure sane block size */
|
|
if (table->blksize_size < table->hardsect_size) {
|
|
table->err_msg = "block size smaller than hardsect size";
|
|
PARSE_ERROR;
|
|
}
|
|
|
|
}
|
|
|
|
#undef PARSE_ERROR
|
|
|
|
if (!was_error) {
|
|
if (dm_table_complete(table) == 0)
|
|
return table;
|
|
}
|
|
|
|
dm_put_table(table);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* convert the text in txt to an unsigned int,
|
|
* returns 0 on failure.
|
|
*/
|
|
int dm_get_number(struct text_region *txt, unsigned int *n)
|
|
{
|
|
char *ptr;
|
|
|
|
dm_eat_space(txt);
|
|
if (dm_empty_tok(txt))
|
|
return 0;
|
|
|
|
*n = simple_strtoul(txt->b, &ptr, 10);
|
|
if (ptr == txt->b)
|
|
return 0;
|
|
|
|
txt->b = ptr;
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* extracts text up to the next '\n'.
|
|
*/
|
|
int dm_get_line(struct text_region *txt, struct text_region *line)
|
|
{
|
|
const char *ptr;
|
|
|
|
dm_eat_space(txt);
|
|
if (dm_empty_tok(txt))
|
|
return 0;
|
|
|
|
ptr = line->b = txt->b;
|
|
while((ptr != txt->e) && (*ptr != '\n'))
|
|
ptr++;
|
|
|
|
txt->b = line->e = ptr;
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* extracts the next non-whitespace token from the file.
|
|
*/
|
|
int dm_get_word(struct text_region *txt, struct text_region *word)
|
|
{
|
|
const char *ptr;
|
|
|
|
dm_eat_space(txt);
|
|
|
|
if (dm_empty_tok(txt))
|
|
return 0;
|
|
|
|
word->b = txt->b;
|
|
for (ptr = word->b = txt->b;
|
|
ptr != txt->e && !isspace((int) *ptr); ptr++)
|
|
;
|
|
|
|
word->e = txt->b = ptr;
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* copy a text region into a traditional C str.
|
|
*/
|
|
void dm_txt_copy(char *dest, size_t max, struct text_region *txt)
|
|
{
|
|
size_t len = txt->e - txt->b;
|
|
if (len > --max)
|
|
len = max;
|
|
strncpy(dest, txt->b, len);
|
|
dest[len] = '\0';
|
|
}
|
|
|
|
/*
|
|
* skip leading whitespace
|
|
*/
|
|
void dm_eat_space(struct text_region *txt)
|
|
{
|
|
while(txt->b != txt->e && isspace((int) *txt->b))
|
|
(txt->b)++;
|
|
}
|
|
|
|
|
|
EXPORT_SYMBOL(dm_get_number);
|
|
EXPORT_SYMBOL(dm_get_word);
|
|
EXPORT_SYMBOL(dm_txt_copy);
|
|
EXPORT_SYMBOL(dm_eat_space);
|