346 lines
9.1 KiB
C
346 lines
9.1 KiB
C
|
/*
|
||
|
* Copyright (c) International Business Machines Corp., 2006
|
||
|
* Copyright (C) 2008 Nokia Corporation
|
||
|
*
|
||
|
* This program 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 of the License, or
|
||
|
* (at your option) any later version.
|
||
|
*
|
||
|
* This program 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 this program; if not, write to the Free Software
|
||
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* Generating UBI images.
|
||
|
*
|
||
|
* Authors: Oliver Lohmann
|
||
|
* Artem Bityutskiy
|
||
|
*/
|
||
|
|
||
|
#define PROGRAM_NAME "libubigen"
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
#include <stdint.h>
|
||
|
#include <unistd.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#include <mtd/ubi-media.h>
|
||
|
#include <mtd_swab.h>
|
||
|
#include <libubigen.h>
|
||
|
#include <crc32.h>
|
||
|
#include "common.h"
|
||
|
#include "include/ubi-fastmap.h"
|
||
|
|
||
|
void ubigen_info_init(struct ubigen_info *ui, int peb_size, int min_io_size,
|
||
|
int subpage_size, int vid_hdr_offs, int ubi_ver,
|
||
|
uint32_t image_seq, int max_leb_count)
|
||
|
{
|
||
|
if (!vid_hdr_offs) {
|
||
|
vid_hdr_offs = UBI_EC_HDR_SIZE + subpage_size - 1;
|
||
|
vid_hdr_offs /= subpage_size;
|
||
|
vid_hdr_offs *= subpage_size;
|
||
|
}
|
||
|
|
||
|
ui->peb_size = peb_size;
|
||
|
ui->min_io_size = min_io_size;
|
||
|
ui->vid_hdr_offs = vid_hdr_offs;
|
||
|
ui->data_offs = vid_hdr_offs + UBI_VID_HDR_SIZE + min_io_size - 1;
|
||
|
ui->data_offs /= min_io_size;
|
||
|
ui->data_offs *= min_io_size;
|
||
|
ui->leb_size = peb_size - ui->data_offs;
|
||
|
ui->ubi_ver = ubi_ver;
|
||
|
ui->image_seq = image_seq;
|
||
|
|
||
|
ui->max_volumes = ui->leb_size / UBI_VTBL_RECORD_SIZE;
|
||
|
if (ui->max_volumes > UBI_MAX_VOLUMES)
|
||
|
ui->max_volumes = UBI_MAX_VOLUMES;
|
||
|
ui->vtbl_size = ui->max_volumes * UBI_VTBL_RECORD_SIZE;
|
||
|
ui->max_leb_count = (max_leb_count > 0 ? max_leb_count : 0);
|
||
|
}
|
||
|
|
||
|
struct ubi_vtbl_record *ubigen_create_empty_vtbl(const struct ubigen_info *ui)
|
||
|
{
|
||
|
struct ubi_vtbl_record *vtbl;
|
||
|
int i;
|
||
|
|
||
|
vtbl = calloc(1, ui->vtbl_size);
|
||
|
if (!vtbl) {
|
||
|
sys_errmsg("cannot allocate %d bytes of memory", ui->vtbl_size);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < ui->max_volumes; i++) {
|
||
|
uint32_t crc = mtd_crc32(UBI_CRC32_INIT, &vtbl[i],
|
||
|
UBI_VTBL_RECORD_SIZE_CRC);
|
||
|
vtbl[i].crc = cpu_to_be32(crc);
|
||
|
}
|
||
|
|
||
|
return vtbl;
|
||
|
}
|
||
|
|
||
|
int ubigen_add_volume(const struct ubigen_info *ui,
|
||
|
struct ubigen_vol_info *vi,
|
||
|
struct ubi_vtbl_record *vtbl)
|
||
|
{
|
||
|
struct ubi_vtbl_record *vtbl_rec = &vtbl[vi->id];
|
||
|
uint32_t tmp;
|
||
|
|
||
|
if (vi->id >= ui->max_volumes) {
|
||
|
errmsg("too high volume id %d, max. volumes is %d",
|
||
|
vi->id, ui->max_volumes);
|
||
|
errno = EINVAL;
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
if (vi->alignment >= ui->leb_size) {
|
||
|
errmsg("too large alignment %d, max is %d (LEB size)",
|
||
|
vi->alignment, ui->leb_size);
|
||
|
errno = EINVAL;
|
||
|
return -1;
|
||
|
}
|
||
|
memset(vtbl_rec, 0, sizeof(struct ubi_vtbl_record));
|
||
|
tmp = (vi->bytes + ui->leb_size - 1) / ui->leb_size;
|
||
|
vtbl_rec->reserved_pebs = cpu_to_be32(tmp);
|
||
|
vtbl_rec->alignment = cpu_to_be32(vi->alignment);
|
||
|
vtbl_rec->vol_type = vi->type;
|
||
|
tmp = ui->leb_size % vi->alignment;
|
||
|
vtbl_rec->data_pad = cpu_to_be32(tmp);
|
||
|
vtbl_rec->flags = vi->flags;
|
||
|
|
||
|
memcpy(vtbl_rec->name, vi->name, vi->name_len);
|
||
|
vtbl_rec->name[vi->name_len] = '\0';
|
||
|
vtbl_rec->name_len = cpu_to_be16(vi->name_len);
|
||
|
|
||
|
tmp = mtd_crc32(UBI_CRC32_INIT, vtbl_rec, UBI_VTBL_RECORD_SIZE_CRC);
|
||
|
vtbl_rec->crc = cpu_to_be32(tmp);
|
||
|
vi->reserved_pebs = 0;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void ubigen_init_ec_hdr(const struct ubigen_info *ui,
|
||
|
struct ubi_ec_hdr *hdr, long long ec)
|
||
|
{
|
||
|
uint32_t crc;
|
||
|
|
||
|
memset(hdr, 0, sizeof(struct ubi_ec_hdr));
|
||
|
|
||
|
hdr->magic = cpu_to_be32(UBI_EC_HDR_MAGIC);
|
||
|
hdr->version = ui->ubi_ver;
|
||
|
hdr->ec = cpu_to_be64(ec);
|
||
|
hdr->vid_hdr_offset = cpu_to_be32(ui->vid_hdr_offs);
|
||
|
hdr->data_offset = cpu_to_be32(ui->data_offs);
|
||
|
hdr->image_seq = cpu_to_be32(ui->image_seq);
|
||
|
|
||
|
crc = mtd_crc32(UBI_CRC32_INIT, hdr, UBI_EC_HDR_SIZE_CRC);
|
||
|
hdr->hdr_crc = cpu_to_be32(crc);
|
||
|
}
|
||
|
|
||
|
void ubigen_init_vid_hdr(const struct ubigen_info *ui,
|
||
|
const struct ubigen_vol_info *vi,
|
||
|
struct ubi_vid_hdr *hdr, int lnum,
|
||
|
const void *data, int data_size)
|
||
|
{
|
||
|
uint32_t crc;
|
||
|
|
||
|
memset(hdr, 0, sizeof(struct ubi_vid_hdr));
|
||
|
|
||
|
hdr->magic = cpu_to_be32(UBI_VID_HDR_MAGIC);
|
||
|
hdr->version = ui->ubi_ver;
|
||
|
hdr->vol_type = vi->type;
|
||
|
hdr->vol_id = cpu_to_be32(vi->id);
|
||
|
hdr->lnum = cpu_to_be32(lnum);
|
||
|
hdr->data_pad = cpu_to_be32(vi->data_pad);
|
||
|
hdr->compat = vi->compat;
|
||
|
|
||
|
if (vi->type == UBI_VID_STATIC) {
|
||
|
hdr->data_size = cpu_to_be32(data_size);
|
||
|
hdr->used_ebs = cpu_to_be32(vi->used_ebs);
|
||
|
crc = mtd_crc32(UBI_CRC32_INIT, data, data_size);
|
||
|
hdr->data_crc = cpu_to_be32(crc);
|
||
|
}
|
||
|
|
||
|
crc = mtd_crc32(UBI_CRC32_INIT, hdr, UBI_VID_HDR_SIZE_CRC);
|
||
|
hdr->hdr_crc = cpu_to_be32(crc);
|
||
|
}
|
||
|
|
||
|
int ubigen_write_volume(const struct ubigen_info *ui,
|
||
|
struct ubigen_vol_info *vi, long long ec,
|
||
|
long long bytes, int in, int out, struct list_head *used,
|
||
|
int *used_cnt)
|
||
|
{
|
||
|
int len = vi->usable_leb_size, rd, lnum = 0;
|
||
|
char *inbuf, *outbuf;
|
||
|
struct ubi_wl_peb *new_peb;
|
||
|
|
||
|
if (vi->id >= ui->max_volumes) {
|
||
|
errmsg("too high volume id %d, max. volumes is %d",
|
||
|
vi->id, ui->max_volumes);
|
||
|
errno = EINVAL;
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
if (vi->alignment >= ui->leb_size) {
|
||
|
errmsg("too large alignment %d, max is %d (LEB size)",
|
||
|
vi->alignment, ui->leb_size);
|
||
|
errno = EINVAL;
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
inbuf = malloc(ui->leb_size);
|
||
|
if (!inbuf)
|
||
|
return sys_errmsg("cannot allocate %d bytes of memory",
|
||
|
ui->leb_size);
|
||
|
outbuf = malloc(ui->peb_size);
|
||
|
if (!outbuf) {
|
||
|
sys_errmsg("cannot allocate %d bytes of memory", ui->peb_size);
|
||
|
goto out_free;
|
||
|
}
|
||
|
|
||
|
memset(outbuf, 0xFF, ui->data_offs);
|
||
|
ubigen_init_ec_hdr(ui, (struct ubi_ec_hdr *)outbuf, ec);
|
||
|
|
||
|
while (bytes) {
|
||
|
int l;
|
||
|
struct ubi_vid_hdr *vid_hdr;
|
||
|
|
||
|
if (bytes < len)
|
||
|
len = bytes;
|
||
|
bytes -= len;
|
||
|
|
||
|
l = len;
|
||
|
do {
|
||
|
rd = read(in, inbuf + len - l, l);
|
||
|
if (rd != l) {
|
||
|
sys_errmsg("cannot read %d bytes from the input file", l);
|
||
|
goto out_free1;
|
||
|
}
|
||
|
|
||
|
l -= rd;
|
||
|
} while (l);
|
||
|
|
||
|
vid_hdr = (struct ubi_vid_hdr *)(&outbuf[ui->vid_hdr_offs]);
|
||
|
ubigen_init_vid_hdr(ui, vi, vid_hdr, lnum, inbuf, len);
|
||
|
|
||
|
memcpy(outbuf + ui->data_offs, inbuf, len);
|
||
|
memset(outbuf + ui->data_offs + len, 0xFF,
|
||
|
ui->peb_size - ui->data_offs - len);
|
||
|
|
||
|
new_peb = malloc(sizeof(*new_peb));
|
||
|
if (!new_peb) {
|
||
|
sys_errmsg("mem allocation failed");
|
||
|
goto out_free1;
|
||
|
}
|
||
|
new_peb->pnum = (*used_cnt)++;
|
||
|
new_peb->ec = ec;
|
||
|
list_add_tail(&new_peb->list, used);
|
||
|
if (vi->pebs)
|
||
|
vi->pebs[lnum] = new_peb->pnum;
|
||
|
vi->reserved_pebs++;
|
||
|
if (write(out, outbuf, ui->peb_size) != ui->peb_size) {
|
||
|
sys_errmsg("cannot write %d bytes to the output file", ui->peb_size);
|
||
|
goto out_free1;
|
||
|
}
|
||
|
|
||
|
lnum += 1;
|
||
|
}
|
||
|
free(outbuf);
|
||
|
free(inbuf);
|
||
|
return 0;
|
||
|
|
||
|
out_free1:
|
||
|
free(outbuf);
|
||
|
out_free:
|
||
|
free(inbuf);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
int ubigen_write_layout_vol(const struct ubigen_info *ui, int peb1, int peb2,
|
||
|
long long ec1, long long ec2,
|
||
|
struct ubi_vtbl_record *vtbl, int fd,
|
||
|
struct ubigen_vol_info *vi_layout)
|
||
|
{
|
||
|
int ret;
|
||
|
struct ubigen_vol_info *vi;
|
||
|
char *outbuf;
|
||
|
struct ubi_vid_hdr *vid_hdr;
|
||
|
off_t seek;
|
||
|
|
||
|
if (vi_layout) {
|
||
|
vi = vi_layout;
|
||
|
} else {
|
||
|
vi = malloc(sizeof(*vi));
|
||
|
if (!vi)
|
||
|
return sys_errmsg("failed to allocate %d bytes",
|
||
|
sizeof(*vi));
|
||
|
}
|
||
|
vi->bytes = ui->leb_size * UBI_LAYOUT_VOLUME_EBS;
|
||
|
vi->id = UBI_LAYOUT_VOLUME_ID;
|
||
|
vi->alignment = UBI_LAYOUT_VOLUME_ALIGN;
|
||
|
vi->data_pad = ui->leb_size % UBI_LAYOUT_VOLUME_ALIGN;
|
||
|
vi->usable_leb_size = ui->leb_size - vi->data_pad;
|
||
|
vi->data_pad = ui->leb_size - vi->usable_leb_size;
|
||
|
vi->type = UBI_LAYOUT_VOLUME_TYPE;
|
||
|
vi->name = UBI_LAYOUT_VOLUME_NAME;
|
||
|
vi->name_len = strlen(UBI_LAYOUT_VOLUME_NAME);
|
||
|
vi->compat = UBI_LAYOUT_VOLUME_COMPAT;
|
||
|
|
||
|
outbuf = malloc(ui->peb_size);
|
||
|
if (!outbuf) {
|
||
|
if (!vi_layout)
|
||
|
free(vi);
|
||
|
return sys_errmsg("failed to allocate %d bytes",
|
||
|
ui->peb_size);
|
||
|
}
|
||
|
|
||
|
memset(outbuf, 0xFF, ui->data_offs);
|
||
|
vid_hdr = (struct ubi_vid_hdr *)(&outbuf[ui->vid_hdr_offs]);
|
||
|
memcpy(outbuf + ui->data_offs, vtbl, ui->vtbl_size);
|
||
|
memset(outbuf + ui->data_offs + ui->vtbl_size, 0xFF,
|
||
|
ui->peb_size - ui->data_offs - ui->vtbl_size);
|
||
|
|
||
|
seek = (off_t) peb1 * ui->peb_size;
|
||
|
if (lseek(fd, seek, SEEK_SET) != seek) {
|
||
|
sys_errmsg("cannot seek output file");
|
||
|
ret = -1;
|
||
|
goto out_free;
|
||
|
}
|
||
|
|
||
|
ubigen_init_ec_hdr(ui, (struct ubi_ec_hdr *)outbuf, ec1);
|
||
|
ubigen_init_vid_hdr(ui, vi, vid_hdr, 0, NULL, 0);
|
||
|
ret = write(fd, outbuf, ui->peb_size);
|
||
|
if (ret != ui->peb_size) {
|
||
|
sys_errmsg("cannot write %d bytes", ui->peb_size);
|
||
|
ret = -1;
|
||
|
goto out_free;
|
||
|
}
|
||
|
|
||
|
seek = (off_t) peb2 * ui->peb_size;
|
||
|
if (lseek(fd, seek, SEEK_SET) != seek) {
|
||
|
sys_errmsg("cannot seek output file");
|
||
|
ret = -1;
|
||
|
goto out_free;
|
||
|
}
|
||
|
ubigen_init_ec_hdr(ui, (struct ubi_ec_hdr *)outbuf, ec2);
|
||
|
ubigen_init_vid_hdr(ui, vi, vid_hdr, 1, NULL, 0);
|
||
|
ret = write(fd, outbuf, ui->peb_size);
|
||
|
if (ret != ui->peb_size) {
|
||
|
sys_errmsg("cannot write %d bytes", ui->peb_size);
|
||
|
ret = -1;
|
||
|
goto out_free;
|
||
|
}
|
||
|
|
||
|
ret = 0;
|
||
|
out_free:
|
||
|
if (!vi_layout)
|
||
|
free(vi);
|
||
|
free(outbuf);
|
||
|
return ret;
|
||
|
}
|