提交 fafdd2bf 编写于 作者: R Richard Weinberger

UBI: Implement UBI_METAONLY

UBI_METAONLY is a new open mode for UBI volumes, it indicates
that only meta data is being changed.
Meta data in terms of UBI volumes means data which is stored in the
UBI volume table but not on the volume itself.
While it does not interfere with UBI_READONLY and UBI_READWRITE
it is not allowed to use UBI_METAONLY together with UBI_EXCLUSIVE.

Cc: Ezequiel Garcia <ezequiel.garcia@free-electrons.com>
Cc: Andrew Murray <amurray@embedded-bits.co.uk>
Signed-off-by: NRichard Weinberger <richard@nod.at>
Tested-by: NGuido Martínez <guido@vanguardiasur.com.ar>
Reviewed-by: NGuido Martínez <guido@vanguardiasur.com.ar>
Tested-by: NChristoph Fritz <chf.fritz@googlemail.com>
Tested-by: NAndrew Murray <amurray@embedded-bits.co.uk>
上级 346be9bc
...@@ -61,13 +61,13 @@ static int get_exclusive(struct ubi_device *ubi, struct ubi_volume_desc *desc) ...@@ -61,13 +61,13 @@ static int get_exclusive(struct ubi_device *ubi, struct ubi_volume_desc *desc)
struct ubi_volume *vol = desc->vol; struct ubi_volume *vol = desc->vol;
spin_lock(&vol->ubi->volumes_lock); spin_lock(&vol->ubi->volumes_lock);
users = vol->readers + vol->writers + vol->exclusive; users = vol->readers + vol->writers + vol->exclusive + vol->metaonly;
ubi_assert(users > 0); ubi_assert(users > 0);
if (users > 1) { if (users > 1) {
ubi_err(ubi, "%d users for volume %d", users, vol->vol_id); ubi_err(ubi, "%d users for volume %d", users, vol->vol_id);
err = -EBUSY; err = -EBUSY;
} else { } else {
vol->readers = vol->writers = 0; vol->readers = vol->writers = vol->metaonly = 0;
vol->exclusive = 1; vol->exclusive = 1;
err = desc->mode; err = desc->mode;
desc->mode = UBI_EXCLUSIVE; desc->mode = UBI_EXCLUSIVE;
...@@ -87,13 +87,15 @@ static void revoke_exclusive(struct ubi_volume_desc *desc, int mode) ...@@ -87,13 +87,15 @@ static void revoke_exclusive(struct ubi_volume_desc *desc, int mode)
struct ubi_volume *vol = desc->vol; struct ubi_volume *vol = desc->vol;
spin_lock(&vol->ubi->volumes_lock); spin_lock(&vol->ubi->volumes_lock);
ubi_assert(vol->readers == 0 && vol->writers == 0); ubi_assert(vol->readers == 0 && vol->writers == 0 && vol->metaonly == 0);
ubi_assert(vol->exclusive == 1 && desc->mode == UBI_EXCLUSIVE); ubi_assert(vol->exclusive == 1 && desc->mode == UBI_EXCLUSIVE);
vol->exclusive = 0; vol->exclusive = 0;
if (mode == UBI_READONLY) if (mode == UBI_READONLY)
vol->readers = 1; vol->readers = 1;
else if (mode == UBI_READWRITE) else if (mode == UBI_READWRITE)
vol->writers = 1; vol->writers = 1;
else if (mode == UBI_METAONLY)
vol->metaonly = 1;
else else
vol->exclusive = 1; vol->exclusive = 1;
spin_unlock(&vol->ubi->volumes_lock); spin_unlock(&vol->ubi->volumes_lock);
......
...@@ -137,7 +137,7 @@ struct ubi_volume_desc *ubi_open_volume(int ubi_num, int vol_id, int mode) ...@@ -137,7 +137,7 @@ struct ubi_volume_desc *ubi_open_volume(int ubi_num, int vol_id, int mode)
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
if (mode != UBI_READONLY && mode != UBI_READWRITE && if (mode != UBI_READONLY && mode != UBI_READWRITE &&
mode != UBI_EXCLUSIVE) mode != UBI_EXCLUSIVE && mode != UBI_METAONLY)
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
/* /*
...@@ -182,10 +182,17 @@ struct ubi_volume_desc *ubi_open_volume(int ubi_num, int vol_id, int mode) ...@@ -182,10 +182,17 @@ struct ubi_volume_desc *ubi_open_volume(int ubi_num, int vol_id, int mode)
break; break;
case UBI_EXCLUSIVE: case UBI_EXCLUSIVE:
if (vol->exclusive || vol->writers || vol->readers) if (vol->exclusive || vol->writers || vol->readers ||
vol->metaonly)
goto out_unlock; goto out_unlock;
vol->exclusive = 1; vol->exclusive = 1;
break; break;
case UBI_METAONLY:
if (vol->metaonly || vol->exclusive)
goto out_unlock;
vol->metaonly = 1;
break;
} }
get_device(&vol->dev); get_device(&vol->dev);
vol->ref_count += 1; vol->ref_count += 1;
...@@ -343,6 +350,10 @@ void ubi_close_volume(struct ubi_volume_desc *desc) ...@@ -343,6 +350,10 @@ void ubi_close_volume(struct ubi_volume_desc *desc)
break; break;
case UBI_EXCLUSIVE: case UBI_EXCLUSIVE:
vol->exclusive = 0; vol->exclusive = 0;
break;
case UBI_METAONLY:
vol->metaonly = 0;
break;
} }
vol->ref_count -= 1; vol->ref_count -= 1;
spin_unlock(&ubi->volumes_lock); spin_unlock(&ubi->volumes_lock);
......
...@@ -261,6 +261,7 @@ struct ubi_fm_pool { ...@@ -261,6 +261,7 @@ struct ubi_fm_pool {
* @readers: number of users holding this volume in read-only mode * @readers: number of users holding this volume in read-only mode
* @writers: number of users holding this volume in read-write mode * @writers: number of users holding this volume in read-write mode
* @exclusive: whether somebody holds this volume in exclusive mode * @exclusive: whether somebody holds this volume in exclusive mode
* @metaonly: whether somebody is altering only meta data of this volume
* *
* @reserved_pebs: how many physical eraseblocks are reserved for this volume * @reserved_pebs: how many physical eraseblocks are reserved for this volume
* @vol_type: volume type (%UBI_DYNAMIC_VOLUME or %UBI_STATIC_VOLUME) * @vol_type: volume type (%UBI_DYNAMIC_VOLUME or %UBI_STATIC_VOLUME)
...@@ -309,6 +310,7 @@ struct ubi_volume { ...@@ -309,6 +310,7 @@ struct ubi_volume {
int readers; int readers;
int writers; int writers;
int exclusive; int exclusive;
int metaonly;
int reserved_pebs; int reserved_pebs;
int vol_type; int vol_type;
...@@ -339,7 +341,8 @@ struct ubi_volume { ...@@ -339,7 +341,8 @@ struct ubi_volume {
/** /**
* struct ubi_volume_desc - UBI volume descriptor returned when it is opened. * struct ubi_volume_desc - UBI volume descriptor returned when it is opened.
* @vol: reference to the corresponding volume description object * @vol: reference to the corresponding volume description object
* @mode: open mode (%UBI_READONLY, %UBI_READWRITE, or %UBI_EXCLUSIVE) * @mode: open mode (%UBI_READONLY, %UBI_READWRITE, %UBI_EXCLUSIVE
* or %UBI_METAONLY)
*/ */
struct ubi_volume_desc { struct ubi_volume_desc {
struct ubi_volume *vol; struct ubi_volume *vol;
...@@ -390,7 +393,8 @@ struct ubi_debug_info { ...@@ -390,7 +393,8 @@ struct ubi_debug_info {
* @volumes_lock: protects @volumes, @rsvd_pebs, @avail_pebs, beb_rsvd_pebs, * @volumes_lock: protects @volumes, @rsvd_pebs, @avail_pebs, beb_rsvd_pebs,
* @beb_rsvd_level, @bad_peb_count, @good_peb_count, @vol_count, * @beb_rsvd_level, @bad_peb_count, @good_peb_count, @vol_count,
* @vol->readers, @vol->writers, @vol->exclusive, * @vol->readers, @vol->writers, @vol->exclusive,
* @vol->ref_count, @vol->mapping and @vol->eba_tbl. * @vol->metaonly, @vol->ref_count, @vol->mapping and
* @vol->eba_tbl.
* @ref_count: count of references on the UBI device * @ref_count: count of references on the UBI device
* @image_seq: image sequence number recorded on EC headers * @image_seq: image sequence number recorded on EC headers
* *
......
...@@ -34,11 +34,14 @@ ...@@ -34,11 +34,14 @@
* UBI_READONLY: read-only mode * UBI_READONLY: read-only mode
* UBI_READWRITE: read-write mode * UBI_READWRITE: read-write mode
* UBI_EXCLUSIVE: exclusive mode * UBI_EXCLUSIVE: exclusive mode
* UBI_METAONLY: modify only the volume meta-data,
* i.e. the data stored in the volume table, but not in any of volume LEBs.
*/ */
enum { enum {
UBI_READONLY = 1, UBI_READONLY = 1,
UBI_READWRITE, UBI_READWRITE,
UBI_EXCLUSIVE UBI_EXCLUSIVE,
UBI_METAONLY
}; };
/** /**
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册