Block patches for 2.0.0-rc1
-----BEGIN PGP SIGNATURE----- Version: GnuPG v2.0.22 (GNU/Linux) iQIcBAABAgAGBQJTKZWfAAoJEH8JsnLIjy/WPGkP/1DibTfX6kXXwRNtt4GDsQFL 1l5lyazUlYP9q5Y7nOWcGrshzCZUX0SmktOble1TlWWlKK7g0iPcSnC6uq7bxS3p oqg3bG6I8rKFmy1/YILwl+Dx8tbA5KarUW8Fv36GFDmR/jn9TEQAjAcJqqBty/lG vVu01rf1dv1qhM37Rd9TZDyXTLTaAE0UeXjnafcppN+P6XRlQkC9JvFWIq+uRWiY oPRgHnLkuffsWuHCt0qqlM4OewIM9f7axX7DQmUiWdRRbxZeWKxQG5E4Tws9RsUa QG7A2NDRmisyry9qaJMKMrnOwBBtJdAGkMqWEynSfY5pampIugNt9dy6A8OP5q9z 3hmZWjulxN8ZfFn71ZM4nwC1MeR9WW/AiIOCIOr40u/5Nb+GhGSvEvRvewwKz7j9 nCCIJKlRE6XP88shQMyxUfPKutYc4BeSCMOmpyDa8hVczljLsU8rBGFGWv4mmafM o7KWo5Lea2//Y6t28IacjZ2awSF3Yzc8e95JgxcMap2P/bWSQuC8IhxTj4JGqXgn nx7WV0gvr6N6/wPw3vbhdJzYOY1vdGYcqmlrZGSl9tYYd8iP7VFucsaiML1r8vT+ Ye96EuhGVUo0poG0mQzCZSv3KjU9tlr4P+KDMGPEbDAEyFDywySjcHeWuiymHRGL BfNLCl95GuQN3y2iCkNo =W6Lu -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/kevin/tags/for-upstream' into staging Block patches for 2.0.0-rc1 # gpg: Signature made Wed 19 Mar 2014 13:03:27 GMT using RSA key ID C88F2FD6 # gpg: Good signature from "Kevin Wolf <kwolf@redhat.com>" * remotes/kevin/tags/for-upstream: dataplane: fix implicit IOThread refcount block/nfs: report errors from libnfs block/nfs: bump libnfs requirement to 1.9.3 qcow2: Fix fail path in realloc_refcount_block() qcow2: Correct comment for realloc_refcount_block() qemu-io: Extended "--cmd" description in usage text qemu-io-cmds: Fixed typo in example for writev. block: Add error handling to bdrv_invalidate_cache() Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
c1b94a0ed2
28
block.c
28
block.c
@ -4781,27 +4781,43 @@ flush_parent:
|
|||||||
return bdrv_co_flush(bs->file);
|
return bdrv_co_flush(bs->file);
|
||||||
}
|
}
|
||||||
|
|
||||||
void bdrv_invalidate_cache(BlockDriverState *bs)
|
void bdrv_invalidate_cache(BlockDriverState *bs, Error **errp)
|
||||||
{
|
{
|
||||||
|
Error *local_err = NULL;
|
||||||
|
int ret;
|
||||||
|
|
||||||
if (!bs->drv) {
|
if (!bs->drv) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bs->drv->bdrv_invalidate_cache) {
|
if (bs->drv->bdrv_invalidate_cache) {
|
||||||
bs->drv->bdrv_invalidate_cache(bs);
|
bs->drv->bdrv_invalidate_cache(bs, &local_err);
|
||||||
} else if (bs->file) {
|
} else if (bs->file) {
|
||||||
bdrv_invalidate_cache(bs->file);
|
bdrv_invalidate_cache(bs->file, &local_err);
|
||||||
|
}
|
||||||
|
if (local_err) {
|
||||||
|
error_propagate(errp, local_err);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
refresh_total_sectors(bs, bs->total_sectors);
|
ret = refresh_total_sectors(bs, bs->total_sectors);
|
||||||
|
if (ret < 0) {
|
||||||
|
error_setg_errno(errp, -ret, "Could not refresh total sector count");
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void bdrv_invalidate_cache_all(void)
|
void bdrv_invalidate_cache_all(Error **errp)
|
||||||
{
|
{
|
||||||
BlockDriverState *bs;
|
BlockDriverState *bs;
|
||||||
|
Error *local_err = NULL;
|
||||||
|
|
||||||
QTAILQ_FOREACH(bs, &bdrv_states, device_list) {
|
QTAILQ_FOREACH(bs, &bdrv_states, device_list) {
|
||||||
bdrv_invalidate_cache(bs);
|
bdrv_invalidate_cache(bs, &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
error_propagate(errp, local_err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -112,6 +112,9 @@ nfs_co_generic_cb(int ret, struct nfs_context *nfs, void *data,
|
|||||||
if (task->ret == 0 && task->st) {
|
if (task->ret == 0 && task->st) {
|
||||||
memcpy(task->st, data, sizeof(struct stat));
|
memcpy(task->st, data, sizeof(struct stat));
|
||||||
}
|
}
|
||||||
|
if (task->ret < 0) {
|
||||||
|
error_report("NFS Error: %s", nfs_get_error(nfs));
|
||||||
|
}
|
||||||
if (task->co) {
|
if (task->co) {
|
||||||
task->bh = qemu_bh_new(nfs_co_generic_bh_cb, task);
|
task->bh = qemu_bh_new(nfs_co_generic_bh_cb, task);
|
||||||
qemu_bh_schedule(task->bh);
|
qemu_bh_schedule(task->bh);
|
||||||
|
@ -1383,7 +1383,7 @@ static int write_reftable_entry(BlockDriverState *bs, int rt_index)
|
|||||||
* does _not_ decrement the reference count for the currently occupied cluster.
|
* does _not_ decrement the reference count for the currently occupied cluster.
|
||||||
*
|
*
|
||||||
* This function prints an informative message to stderr on error (and returns
|
* This function prints an informative message to stderr on error (and returns
|
||||||
* -errno); on success, 0 is returned.
|
* -errno); on success, the offset of the newly allocated cluster is returned.
|
||||||
*/
|
*/
|
||||||
static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
|
static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
|
||||||
uint64_t offset)
|
uint64_t offset)
|
||||||
@ -1399,14 +1399,14 @@ static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
|
|||||||
fprintf(stderr, "Could not allocate new cluster: %s\n",
|
fprintf(stderr, "Could not allocate new cluster: %s\n",
|
||||||
strerror(-new_offset));
|
strerror(-new_offset));
|
||||||
ret = new_offset;
|
ret = new_offset;
|
||||||
goto fail;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* fetch current refcount block content */
|
/* fetch current refcount block content */
|
||||||
ret = qcow2_cache_get(bs, s->refcount_block_cache, offset, &refcount_block);
|
ret = qcow2_cache_get(bs, s->refcount_block_cache, offset, &refcount_block);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
fprintf(stderr, "Could not fetch refcount block: %s\n", strerror(-ret));
|
fprintf(stderr, "Could not fetch refcount block: %s\n", strerror(-ret));
|
||||||
goto fail;
|
goto fail_free_cluster;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* new block has not yet been entered into refcount table, therefore it is
|
/* new block has not yet been entered into refcount table, therefore it is
|
||||||
@ -1417,8 +1417,7 @@ static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
|
|||||||
"check failed: %s\n", strerror(-ret));
|
"check failed: %s\n", strerror(-ret));
|
||||||
/* the image will be marked corrupt, so don't even attempt on freeing
|
/* the image will be marked corrupt, so don't even attempt on freeing
|
||||||
* the cluster */
|
* the cluster */
|
||||||
new_offset = 0;
|
goto done;
|
||||||
goto fail;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* write to new block */
|
/* write to new block */
|
||||||
@ -1426,7 +1425,7 @@ static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
|
|||||||
s->cluster_sectors);
|
s->cluster_sectors);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
fprintf(stderr, "Could not write refcount block: %s\n", strerror(-ret));
|
fprintf(stderr, "Could not write refcount block: %s\n", strerror(-ret));
|
||||||
goto fail;
|
goto fail_free_cluster;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* update refcount table */
|
/* update refcount table */
|
||||||
@ -1436,24 +1435,27 @@ static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
|
|||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
fprintf(stderr, "Could not update refcount table: %s\n",
|
fprintf(stderr, "Could not update refcount table: %s\n",
|
||||||
strerror(-ret));
|
strerror(-ret));
|
||||||
goto fail;
|
goto fail_free_cluster;
|
||||||
}
|
}
|
||||||
|
|
||||||
fail:
|
goto done;
|
||||||
if (new_offset && (ret < 0)) {
|
|
||||||
qcow2_free_clusters(bs, new_offset, s->cluster_size,
|
fail_free_cluster:
|
||||||
QCOW2_DISCARD_ALWAYS);
|
qcow2_free_clusters(bs, new_offset, s->cluster_size, QCOW2_DISCARD_OTHER);
|
||||||
}
|
|
||||||
|
done:
|
||||||
if (refcount_block) {
|
if (refcount_block) {
|
||||||
if (ret < 0) {
|
/* This should never fail, as it would only do so if the given refcount
|
||||||
qcow2_cache_put(bs, s->refcount_block_cache, &refcount_block);
|
* block cannot be found in the cache. As this is impossible as long as
|
||||||
} else {
|
* there are no bugs, assert the success. */
|
||||||
ret = qcow2_cache_put(bs, s->refcount_block_cache, &refcount_block);
|
int tmp = qcow2_cache_put(bs, s->refcount_block_cache, &refcount_block);
|
||||||
}
|
assert(tmp == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
return new_offset;
|
return new_offset;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1156,7 +1156,7 @@ static void qcow2_close(BlockDriverState *bs)
|
|||||||
qcow2_free_snapshots(bs);
|
qcow2_free_snapshots(bs);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void qcow2_invalidate_cache(BlockDriverState *bs)
|
static void qcow2_invalidate_cache(BlockDriverState *bs, Error **errp)
|
||||||
{
|
{
|
||||||
BDRVQcowState *s = bs->opaque;
|
BDRVQcowState *s = bs->opaque;
|
||||||
int flags = s->flags;
|
int flags = s->flags;
|
||||||
@ -1164,6 +1164,8 @@ static void qcow2_invalidate_cache(BlockDriverState *bs)
|
|||||||
AES_KEY aes_decrypt_key;
|
AES_KEY aes_decrypt_key;
|
||||||
uint32_t crypt_method = 0;
|
uint32_t crypt_method = 0;
|
||||||
QDict *options;
|
QDict *options;
|
||||||
|
Error *local_err = NULL;
|
||||||
|
int ret;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Backing files are read-only which makes all of their metadata immutable,
|
* Backing files are read-only which makes all of their metadata immutable,
|
||||||
@ -1178,11 +1180,25 @@ static void qcow2_invalidate_cache(BlockDriverState *bs)
|
|||||||
|
|
||||||
qcow2_close(bs);
|
qcow2_close(bs);
|
||||||
|
|
||||||
bdrv_invalidate_cache(bs->file);
|
bdrv_invalidate_cache(bs->file, &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
error_propagate(errp, local_err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
memset(s, 0, sizeof(BDRVQcowState));
|
memset(s, 0, sizeof(BDRVQcowState));
|
||||||
options = qdict_clone_shallow(bs->options);
|
options = qdict_clone_shallow(bs->options);
|
||||||
qcow2_open(bs, options, flags, NULL);
|
|
||||||
|
ret = qcow2_open(bs, options, flags, &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
error_setg(errp, "Could not reopen qcow2 layer: %s",
|
||||||
|
error_get_pretty(local_err));
|
||||||
|
error_free(local_err);
|
||||||
|
return;
|
||||||
|
} else if (ret < 0) {
|
||||||
|
error_setg_errno(errp, -ret, "Could not reopen qcow2 layer");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
QDECREF(options);
|
QDECREF(options);
|
||||||
|
|
||||||
|
21
block/qed.c
21
block/qed.c
@ -1558,16 +1558,31 @@ static int bdrv_qed_change_backing_file(BlockDriverState *bs,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void bdrv_qed_invalidate_cache(BlockDriverState *bs)
|
static void bdrv_qed_invalidate_cache(BlockDriverState *bs, Error **errp)
|
||||||
{
|
{
|
||||||
BDRVQEDState *s = bs->opaque;
|
BDRVQEDState *s = bs->opaque;
|
||||||
|
Error *local_err = NULL;
|
||||||
|
int ret;
|
||||||
|
|
||||||
bdrv_qed_close(bs);
|
bdrv_qed_close(bs);
|
||||||
|
|
||||||
bdrv_invalidate_cache(bs->file);
|
bdrv_invalidate_cache(bs->file, &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
error_propagate(errp, local_err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
memset(s, 0, sizeof(BDRVQEDState));
|
memset(s, 0, sizeof(BDRVQEDState));
|
||||||
bdrv_qed_open(bs, NULL, bs->open_flags, NULL);
|
ret = bdrv_qed_open(bs, NULL, bs->open_flags, &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
error_setg(errp, "Could not reopen qed layer: %s",
|
||||||
|
error_get_pretty(local_err));
|
||||||
|
error_free(local_err);
|
||||||
|
return;
|
||||||
|
} else if (ret < 0) {
|
||||||
|
error_setg_errno(errp, -ret, "Could not reopen qed layer");
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int bdrv_qed_check(BlockDriverState *bs, BdrvCheckResult *result,
|
static int bdrv_qed_check(BlockDriverState *bs, BdrvCheckResult *result,
|
||||||
|
@ -625,13 +625,18 @@ static int64_t quorum_getlength(BlockDriverState *bs)
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void quorum_invalidate_cache(BlockDriverState *bs)
|
static void quorum_invalidate_cache(BlockDriverState *bs, Error **errp)
|
||||||
{
|
{
|
||||||
BDRVQuorumState *s = bs->opaque;
|
BDRVQuorumState *s = bs->opaque;
|
||||||
|
Error *local_err = NULL;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; i < s->num_children; i++) {
|
for (i = 0; i < s->num_children; i++) {
|
||||||
bdrv_invalidate_cache(s->bs[i]);
|
bdrv_invalidate_cache(s->bs[i], &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
error_propagate(errp, local_err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
2
configure
vendored
2
configure
vendored
@ -3868,7 +3868,7 @@ fi
|
|||||||
##########################################
|
##########################################
|
||||||
# Do we have libnfs
|
# Do we have libnfs
|
||||||
if test "$libnfs" != "no" ; then
|
if test "$libnfs" != "no" ; then
|
||||||
if $pkg_config --atleast-version=1.9.2 libnfs; then
|
if $pkg_config --atleast-version=1.9.3 libnfs; then
|
||||||
libnfs="yes"
|
libnfs="yes"
|
||||||
libnfs_libs=$($pkg_config --libs libnfs)
|
libnfs_libs=$($pkg_config --libs libnfs)
|
||||||
LIBS="$LIBS $libnfs_libs"
|
LIBS="$LIBS $libnfs_libs"
|
||||||
|
@ -393,7 +393,6 @@ void virtio_blk_data_plane_create(VirtIODevice *vdev, VirtIOBlkConf *blk,
|
|||||||
if (blk->iothread) {
|
if (blk->iothread) {
|
||||||
s->internal_iothread = false;
|
s->internal_iothread = false;
|
||||||
s->iothread = blk->iothread;
|
s->iothread = blk->iothread;
|
||||||
object_ref(OBJECT(s->iothread));
|
|
||||||
} else {
|
} else {
|
||||||
/* Create per-device IOThread if none specified */
|
/* Create per-device IOThread if none specified */
|
||||||
Error *local_err = NULL;
|
Error *local_err = NULL;
|
||||||
@ -408,6 +407,7 @@ void virtio_blk_data_plane_create(VirtIODevice *vdev, VirtIOBlkConf *blk,
|
|||||||
s->iothread = iothread_find(vdev->name);
|
s->iothread = iothread_find(vdev->name);
|
||||||
assert(s->iothread);
|
assert(s->iothread);
|
||||||
}
|
}
|
||||||
|
object_ref(OBJECT(s->iothread));
|
||||||
s->ctx = iothread_get_aio_context(s->iothread);
|
s->ctx = iothread_get_aio_context(s->iothread);
|
||||||
|
|
||||||
/* Prevent block operations that conflict with data plane thread */
|
/* Prevent block operations that conflict with data plane thread */
|
||||||
|
@ -329,8 +329,8 @@ BlockDriverAIOCB *bdrv_aio_ioctl(BlockDriverState *bs,
|
|||||||
BlockDriverCompletionFunc *cb, void *opaque);
|
BlockDriverCompletionFunc *cb, void *opaque);
|
||||||
|
|
||||||
/* Invalidate any cached metadata used by image formats */
|
/* Invalidate any cached metadata used by image formats */
|
||||||
void bdrv_invalidate_cache(BlockDriverState *bs);
|
void bdrv_invalidate_cache(BlockDriverState *bs, Error **errp);
|
||||||
void bdrv_invalidate_cache_all(void);
|
void bdrv_invalidate_cache_all(Error **errp);
|
||||||
|
|
||||||
void bdrv_clear_incoming_migration_all(void);
|
void bdrv_clear_incoming_migration_all(void);
|
||||||
|
|
||||||
|
@ -153,7 +153,7 @@ struct BlockDriver {
|
|||||||
/*
|
/*
|
||||||
* Invalidate any cached meta-data.
|
* Invalidate any cached meta-data.
|
||||||
*/
|
*/
|
||||||
void (*bdrv_invalidate_cache)(BlockDriverState *bs);
|
void (*bdrv_invalidate_cache)(BlockDriverState *bs, Error **errp);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Flushes all data that was already written to the OS all the way down to
|
* Flushes all data that was already written to the OS all the way down to
|
||||||
|
@ -101,6 +101,7 @@ void qemu_start_incoming_migration(const char *uri, Error **errp)
|
|||||||
static void process_incoming_migration_co(void *opaque)
|
static void process_incoming_migration_co(void *opaque)
|
||||||
{
|
{
|
||||||
QEMUFile *f = opaque;
|
QEMUFile *f = opaque;
|
||||||
|
Error *local_err = NULL;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ret = qemu_loadvm_state(f);
|
ret = qemu_loadvm_state(f);
|
||||||
@ -115,7 +116,12 @@ static void process_incoming_migration_co(void *opaque)
|
|||||||
|
|
||||||
bdrv_clear_incoming_migration_all();
|
bdrv_clear_incoming_migration_all();
|
||||||
/* Make sure all file formats flush their mutable metadata */
|
/* Make sure all file formats flush their mutable metadata */
|
||||||
bdrv_invalidate_cache_all();
|
bdrv_invalidate_cache_all(&local_err);
|
||||||
|
if (local_err) {
|
||||||
|
qerror_report_err(local_err);
|
||||||
|
error_free(local_err);
|
||||||
|
exit(EXIT_FAILURE);
|
||||||
|
}
|
||||||
|
|
||||||
if (autostart) {
|
if (autostart) {
|
||||||
vm_start();
|
vm_start();
|
||||||
|
@ -1087,7 +1087,7 @@ writev_help(void)
|
|||||||
" writes a range of bytes from the given offset source from multiple buffers\n"
|
" writes a range of bytes from the given offset source from multiple buffers\n"
|
||||||
"\n"
|
"\n"
|
||||||
" Example:\n"
|
" Example:\n"
|
||||||
" 'write 512 1k 1k' - writes 2 kilobytes at 512 bytes into the open file\n"
|
" 'writev 512 1k 1k' - writes 2 kilobytes at 512 bytes into the open file\n"
|
||||||
"\n"
|
"\n"
|
||||||
" Writes into a segment of the currently open file, using a buffer\n"
|
" Writes into a segment of the currently open file, using a buffer\n"
|
||||||
" filled with a set pattern (0xcdcdcdcd).\n"
|
" filled with a set pattern (0xcdcdcdcd).\n"
|
||||||
|
@ -193,10 +193,11 @@ static const cmdinfo_t quit_cmd = {
|
|||||||
static void usage(const char *name)
|
static void usage(const char *name)
|
||||||
{
|
{
|
||||||
printf(
|
printf(
|
||||||
"Usage: %s [-h] [-V] [-rsnm] [-c cmd] ... [file]\n"
|
"Usage: %s [-h] [-V] [-rsnm] [-c STRING] ... [file]\n"
|
||||||
"QEMU Disk exerciser\n"
|
"QEMU Disk exerciser\n"
|
||||||
"\n"
|
"\n"
|
||||||
" -c, --cmd command to execute\n"
|
" -c, --cmd STRING execute command with its arguments\n"
|
||||||
|
" from the given string\n"
|
||||||
" -r, --read-only export read-only\n"
|
" -r, --read-only export read-only\n"
|
||||||
" -s, --snapshot use snapshot file\n"
|
" -s, --snapshot use snapshot file\n"
|
||||||
" -n, --nocache disable host cache\n"
|
" -n, --nocache disable host cache\n"
|
||||||
@ -207,8 +208,10 @@ static void usage(const char *name)
|
|||||||
" -T, --trace FILE enable trace events listed in the given file\n"
|
" -T, --trace FILE enable trace events listed in the given file\n"
|
||||||
" -h, --help display this help and exit\n"
|
" -h, --help display this help and exit\n"
|
||||||
" -V, --version output version information and exit\n"
|
" -V, --version output version information and exit\n"
|
||||||
|
"\n"
|
||||||
|
"See '%s -c help' for information on available commands."
|
||||||
"\n",
|
"\n",
|
||||||
name);
|
name, name);
|
||||||
}
|
}
|
||||||
|
|
||||||
static char *get_prompt(void)
|
static char *get_prompt(void)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user