1 // -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
2 // vim: ts=8 sw=2 smarttab
4 * Ceph - scalable distributed file system
6 * Copyright (C) 2017 Red Hat Ltd
8 * This is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License version 2.1, as published by the Free Software
11 * Foundation. See file COPYING.
16 #include "OSDMonitor.h"
17 #include "PGMonitor.h"
19 #include "FSCommands.h"
20 #include "MDSMonitor.h"
24 static const string
EXPERIMENTAL_WARNING("Warning! This feature is experimental."
25 "It may cause problems up to and including data loss."
26 "Consult the documentation at ceph.com, and if unsure, do not proceed."
27 "Add --yes-i-really-mean-it if you are certain.");
31 class FlagSetHandler
: public FileSystemCommandHandler
35 : FileSystemCommandHandler("fs flag set")
43 map
<string
, cmd_vartype
> &cmdmap
,
44 std::stringstream
&ss
) override
47 cmd_getval(g_ceph_context
, cmdmap
, "flag_name", flag_name
);
50 cmd_getval(g_ceph_context
, cmdmap
, "val", flag_val
);
53 cmd_getval(g_ceph_context
, cmdmap
, "confirm", confirm
);
55 if (flag_name
== "enable_multiple") {
56 bool flag_bool
= false;
57 int r
= parse_bool(flag_val
, &flag_bool
, ss
);
59 ss
<< "Invalid boolean value '" << flag_val
<< "'";
63 bool jewel
= mon
->get_quorum_con_features() & CEPH_FEATURE_SERVER_JEWEL
;
64 if (flag_bool
&& !jewel
) {
65 ss
<< "Multiple-filesystems are forbidden until all mons are updated";
68 if (confirm
!= "--yes-i-really-mean-it") {
69 ss
<< EXPERIMENTAL_WARNING
;
71 fsmap
.set_enable_multiple(flag_bool
);
74 ss
<< "Unknown flag '" << flag_name
<< "'";
80 class FsNewHandler
: public FileSystemCommandHandler
83 FsNewHandler(Paxos
*paxos
)
84 : FileSystemCommandHandler("fs new"), m_paxos(paxos
)
88 bool batched_propose() override
{
96 map
<string
, cmd_vartype
> &cmdmap
,
97 std::stringstream
&ss
) override
99 assert(m_paxos
->is_plugged());
101 string metadata_name
;
102 cmd_getval(g_ceph_context
, cmdmap
, "metadata", metadata_name
);
103 int64_t metadata
= mon
->osdmon()->osdmap
.lookup_pg_pool_name(metadata_name
);
105 ss
<< "pool '" << metadata_name
<< "' does not exist";
110 cmd_getval(g_ceph_context
,cmdmap
, "force", force_str
);
111 bool force
= (force_str
== "--force");
112 const pool_stat_t
*stat
= mon
->pgservice
->get_pool_stat(metadata
);
114 int64_t metadata_num_objects
= stat
->stats
.sum
.num_objects
;
115 if (!force
&& metadata_num_objects
> 0) {
116 ss
<< "pool '" << metadata_name
117 << "' already contains some objects. Use an empty pool instead.";
123 cmd_getval(g_ceph_context
, cmdmap
, "data", data_name
);
124 int64_t data
= mon
->osdmon()->osdmap
.lookup_pg_pool_name(data_name
);
126 ss
<< "pool '" << data_name
<< "' does not exist";
130 ss
<< "pool '" << data_name
<< "' has id 0, which CephFS does not allow. Use another pool or recreate it to get a non-zero pool id.";
135 cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
);
136 if (fs_name
.empty()) {
137 // Ensure fs name is not empty so that we can implement
138 // commmands that refer to FS by name in future.
139 ss
<< "Filesystem name may not be empty";
143 if (fsmap
.get_filesystem(fs_name
)) {
144 auto fs
= fsmap
.get_filesystem(fs_name
);
145 if (*(fs
->mds_map
.get_data_pools().begin()) == data
146 && fs
->mds_map
.get_metadata_pool() == metadata
) {
147 // Identical FS created already, this is a no-op
148 ss
<< "filesystem '" << fs_name
<< "' already exists";
151 ss
<< "filesystem already exists with name '" << fs_name
<< "'";
156 if (fsmap
.filesystem_count() > 0
157 && !fsmap
.get_enable_multiple()) {
158 ss
<< "Creation of multiple filesystems is disabled. To enable "
159 "this experimental feature, use 'ceph fs flag set enable_multiple "
164 for (auto fs
: fsmap
.get_filesystems()) {
165 const std::vector
<int64_t> &data_pools
= fs
->mds_map
.get_data_pools();
167 if ((std::find(data_pools
.begin(), data_pools
.end(), data
) != data_pools
.end()
168 || fs
->mds_map
.get_metadata_pool() == metadata
)
169 && ((!cmd_getval(g_ceph_context
, cmdmap
, "sure", sure
)
170 || sure
!= "--allow-dangerous-metadata-overlay"))) {
171 ss
<< "Filesystem '" << fs_name
172 << "' is already using one of the specified RADOS pools. This should ONLY be done in emergencies and after careful reading of the documentation. Pass --allow-dangerous-metadata-overlay to permit this.";
177 pg_pool_t
const *data_pool
= mon
->osdmon()->osdmap
.get_pg_pool(data
);
178 assert(data_pool
!= NULL
); // Checked it existed above
179 pg_pool_t
const *metadata_pool
= mon
->osdmon()->osdmap
.get_pg_pool(metadata
);
180 assert(metadata_pool
!= NULL
); // Checked it existed above
182 // we must make these checks before we even allow ourselves to *think*
183 // about requesting a proposal to the osdmonitor and bail out now if
184 // we believe we must. bailing out *after* we request the proposal is
185 // bad business as we could have changed the osdmon's state and ending up
186 // returning an error to the user.
187 int r
= _check_pool(mon
->osdmon()->osdmap
, data
, false, force
, &ss
);
192 r
= _check_pool(mon
->osdmon()->osdmap
, metadata
, true, force
, &ss
);
197 mon
->osdmon()->do_application_enable(data
,
198 pg_pool_t::APPLICATION_NAME_CEPHFS
);
199 mon
->osdmon()->do_application_enable(metadata
,
200 pg_pool_t::APPLICATION_NAME_CEPHFS
);
202 // All checks passed, go ahead and create.
203 fsmap
.create_filesystem(fs_name
, metadata
, data
,
204 mon
->get_quorum_con_features());
205 ss
<< "new fs with metadata pool " << metadata
<< " and data pool " << data
;
213 class SetHandler
: public FileSystemCommandHandler
217 : FileSystemCommandHandler("fs set")
224 map
<string
, cmd_vartype
> &cmdmap
,
225 std::stringstream
&ss
) override
228 if (!cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
) || fs_name
.empty()) {
229 ss
<< "Missing filesystem name";
233 auto fs
= fsmap
.get_filesystem(fs_name
);
235 ss
<< "Not found: '" << fs_name
<< "'";
240 if (!cmd_getval(g_ceph_context
, cmdmap
, "var", var
) || var
.empty()) {
241 ss
<< "Invalid variable";
247 if (!cmd_getval(g_ceph_context
, cmdmap
, "val", val
)) {
250 // we got a string. see if it contains an int.
251 n
= strict_strtoll(val
.c_str(), 10, &interr
);
252 if (var
== "max_mds") {
253 // NOTE: see also "mds set_max_mds", which can modify the same field.
254 if (interr
.length()) {
260 ss
<< "You must specify at least one MDS";
264 if (!fs
->mds_map
.allows_multimds() && n
> fs
->mds_map
.get_max_mds() &&
266 ss
<< "multi-MDS clusters are not enabled; set 'allow_multimds' to enable";
270 ss
<< "may not have more than " << MAX_MDS
<< " MDS ranks";
273 fsmap
.modify_filesystem(
275 [n
](std::shared_ptr
<Filesystem
> fs
)
277 fs
->mds_map
.set_max_mds(n
);
279 } else if (var
== "inline_data") {
280 bool enable_inline
= false;
281 int r
= parse_bool(val
, &enable_inline
, ss
);
288 if (!cmd_getval(g_ceph_context
, cmdmap
, "confirm", confirm
) ||
289 confirm
!= "--yes-i-really-mean-it") {
290 ss
<< EXPERIMENTAL_WARNING
;
293 ss
<< "inline data enabled";
295 fsmap
.modify_filesystem(
297 [](std::shared_ptr
<Filesystem
> fs
)
299 fs
->mds_map
.set_inline_data_enabled(true);
303 CompatSet c
= fsmap
.get_compat();
304 c
.incompat
.insert(MDS_FEATURE_INCOMPAT_INLINE
);
305 fsmap
.update_compat(c
);
307 ss
<< "inline data disabled";
308 fsmap
.modify_filesystem(
310 [](std::shared_ptr
<Filesystem
> fs
)
312 fs
->mds_map
.set_inline_data_enabled(false);
315 } else if (var
== "balancer") {
317 ss
<< "unsetting the metadata load balancer";
319 ss
<< "setting the metadata load balancer to " << val
;
321 fsmap
.modify_filesystem(
323 [val
](std::shared_ptr
<Filesystem
> fs
)
325 fs
->mds_map
.set_balancer(val
);
328 } else if (var
== "max_file_size") {
329 if (interr
.length()) {
330 ss
<< var
<< " requires an integer value";
333 if (n
< CEPH_MIN_STRIPE_UNIT
) {
334 ss
<< var
<< " must at least " << CEPH_MIN_STRIPE_UNIT
;
337 fsmap
.modify_filesystem(
339 [n
](std::shared_ptr
<Filesystem
> fs
)
341 fs
->mds_map
.set_max_filesize(n
);
343 } else if (var
== "allow_new_snaps") {
344 bool enable_snaps
= false;
345 int r
= parse_bool(val
, &enable_snaps
, ss
);
351 fsmap
.modify_filesystem(
353 [](std::shared_ptr
<Filesystem
> fs
)
355 fs
->mds_map
.clear_snaps_allowed();
357 ss
<< "disabled new snapshots";
360 if (!cmd_getval(g_ceph_context
, cmdmap
, "confirm", confirm
) ||
361 confirm
!= "--yes-i-really-mean-it") {
362 ss
<< EXPERIMENTAL_WARNING
;
365 fsmap
.modify_filesystem(
367 [](std::shared_ptr
<Filesystem
> fs
)
369 fs
->mds_map
.set_snaps_allowed();
371 ss
<< "enabled new snapshots";
373 } else if (var
== "allow_multimds") {
374 bool enable_multimds
= false;
375 int r
= parse_bool(val
, &enable_multimds
, ss
);
380 if (!enable_multimds
) {
381 fsmap
.modify_filesystem(fs
->fscid
,
382 [](std::shared_ptr
<Filesystem
> fs
)
384 fs
->mds_map
.clear_multimds_allowed();
386 ss
<< "disallowed increasing the cluster size past 1";
388 fsmap
.modify_filesystem(
390 [](std::shared_ptr
<Filesystem
> fs
)
392 fs
->mds_map
.set_multimds_allowed();
394 ss
<< "enabled creation of more than 1 active MDS";
396 } else if (var
== "allow_dirfrags") {
397 bool enable_dirfrags
= false;
398 int r
= parse_bool(val
, &enable_dirfrags
, ss
);
403 if (!enable_dirfrags
) {
404 fsmap
.modify_filesystem(fs
->fscid
,
405 [](std::shared_ptr
<Filesystem
> fs
)
407 fs
->mds_map
.clear_dirfrags_allowed();
409 ss
<< "disallowed new directory fragmentation";
411 fsmap
.modify_filesystem(
413 [](std::shared_ptr
<Filesystem
> fs
)
415 fs
->mds_map
.set_dirfrags_allowed();
417 ss
<< "enabled directory fragmentation";
419 } else if (var
== "cluster_down") {
420 bool is_down
= false;
421 int r
= parse_bool(val
, &is_down
, ss
);
426 fsmap
.modify_filesystem(
428 [is_down
](std::shared_ptr
<Filesystem
> fs
)
431 fs
->mds_map
.set_flag(CEPH_MDSMAP_DOWN
);
433 fs
->mds_map
.clear_flag(CEPH_MDSMAP_DOWN
);
437 ss
<< "marked " << (is_down
? "down" : "up");
438 } else if (var
== "standby_count_wanted") {
439 if (interr
.length()) {
440 ss
<< var
<< " requires an integer value";
444 ss
<< var
<< " must be non-negative";
447 fsmap
.modify_filesystem(
449 [n
](std::shared_ptr
<Filesystem
> fs
)
451 fs
->mds_map
.set_standby_count_wanted(n
);
454 ss
<< "unknown variable " << var
;
462 class AddDataPoolHandler
: public FileSystemCommandHandler
465 AddDataPoolHandler(Paxos
*paxos
)
466 : FileSystemCommandHandler("fs add_data_pool"), m_paxos(paxos
)
469 bool batched_propose() override
{
477 map
<string
, cmd_vartype
> &cmdmap
,
478 std::stringstream
&ss
) override
480 assert(m_paxos
->is_plugged());
483 cmd_getval(g_ceph_context
, cmdmap
, "pool", poolname
);
486 if (!cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
)
487 || fs_name
.empty()) {
488 ss
<< "Missing filesystem name";
492 auto fs
= fsmap
.get_filesystem(fs_name
);
494 ss
<< "Not found: '" << fs_name
<< "'";
498 int64_t poolid
= mon
->osdmon()->osdmap
.lookup_pg_pool_name(poolname
);
501 poolid
= strict_strtol(poolname
.c_str(), 10, &err
);
503 ss
<< "pool '" << poolname
<< "' does not exist";
508 int r
= _check_pool(mon
->osdmon()->osdmap
, poolid
, false, false, &ss
);
513 // no-op when the data_pool already on fs
514 if (fs
->mds_map
.is_data_pool(poolid
)) {
515 ss
<< "data pool " << poolid
<< " is already on fs " << fs_name
;
519 mon
->osdmon()->do_application_enable(poolid
,
520 pg_pool_t::APPLICATION_NAME_CEPHFS
);
522 fsmap
.modify_filesystem(
524 [poolid
](std::shared_ptr
<Filesystem
> fs
)
526 fs
->mds_map
.add_data_pool(poolid
);
529 ss
<< "added data pool " << poolid
<< " to fsmap";
538 class SetDefaultHandler
: public FileSystemCommandHandler
542 : FileSystemCommandHandler("fs set-default")
549 map
<string
, cmd_vartype
> &cmdmap
,
550 std::stringstream
&ss
) override
553 cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
);
554 auto fs
= fsmap
.get_filesystem(fs_name
);
556 ss
<< "filesystem '" << fs_name
<< "' does not exist";
560 fsmap
.set_legacy_client_fscid(fs
->fscid
);
565 class RemoveFilesystemHandler
: public FileSystemCommandHandler
568 RemoveFilesystemHandler()
569 : FileSystemCommandHandler("fs rm")
576 map
<string
, cmd_vartype
> &cmdmap
,
577 std::stringstream
&ss
) override
579 // Check caller has correctly named the FS to delete
580 // (redundant while there is only one FS, but command
581 // syntax should apply to multi-FS future)
583 cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
);
584 auto fs
= fsmap
.get_filesystem(fs_name
);
586 // Consider absence success to make deletes idempotent
587 ss
<< "filesystem '" << fs_name
<< "' does not exist";
591 // Check that no MDS daemons are active
592 if (fs
->mds_map
.get_num_up_mds() > 0) {
593 ss
<< "all MDS daemons must be inactive before removing filesystem";
597 // Check for confirmation flag
599 cmd_getval(g_ceph_context
, cmdmap
, "sure", sure
);
600 if (sure
!= "--yes-i-really-mean-it") {
601 ss
<< "this is a DESTRUCTIVE operation and will make data in your filesystem permanently" \
602 " inaccessible. Add --yes-i-really-mean-it if you are sure you wish to continue.";
606 if (fsmap
.get_legacy_client_fscid() == fs
->fscid
) {
607 fsmap
.set_legacy_client_fscid(FS_CLUSTER_ID_NONE
);
610 std::vector
<mds_gid_t
> to_fail
;
611 // There may be standby_replay daemons left here
612 for (const auto &i
: fs
->mds_map
.get_mds_info()) {
613 assert(i
.second
.state
== MDSMap::STATE_STANDBY_REPLAY
);
614 to_fail
.push_back(i
.first
);
617 for (const auto &gid
: to_fail
) {
618 // Standby replays don't write, so it isn't important to
619 // wait for an osdmap propose here: ignore return value.
620 mon
->mdsmon()->fail_mds_gid(gid
);
623 fsmap
.erase_filesystem(fs
->fscid
);
629 class ResetFilesystemHandler
: public FileSystemCommandHandler
632 ResetFilesystemHandler()
633 : FileSystemCommandHandler("fs reset")
640 map
<string
, cmd_vartype
> &cmdmap
,
641 std::stringstream
&ss
) override
644 cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
);
645 auto fs
= fsmap
.get_filesystem(fs_name
);
647 ss
<< "filesystem '" << fs_name
<< "' does not exist";
648 // Unlike fs rm, we consider this case an error
652 // Check that no MDS daemons are active
653 if (fs
->mds_map
.get_num_up_mds() > 0) {
654 ss
<< "all MDS daemons must be inactive before resetting filesystem: set the cluster_down flag"
655 " and use `ceph mds fail` to make this so";
659 // Check for confirmation flag
661 cmd_getval(g_ceph_context
, cmdmap
, "sure", sure
);
662 if (sure
!= "--yes-i-really-mean-it") {
663 ss
<< "this is a potentially destructive operation, only for use by experts in disaster recovery. "
664 "Add --yes-i-really-mean-it if you are sure you wish to continue.";
668 fsmap
.reset_filesystem(fs
->fscid
);
674 class RemoveDataPoolHandler
: public FileSystemCommandHandler
677 RemoveDataPoolHandler()
678 : FileSystemCommandHandler("fs rm_data_pool")
685 map
<string
, cmd_vartype
> &cmdmap
,
686 std::stringstream
&ss
) override
689 cmd_getval(g_ceph_context
, cmdmap
, "pool", poolname
);
692 if (!cmd_getval(g_ceph_context
, cmdmap
, "fs_name", fs_name
)
693 || fs_name
.empty()) {
694 ss
<< "Missing filesystem name";
698 auto fs
= fsmap
.get_filesystem(fs_name
);
700 ss
<< "Not found: '" << fs_name
<< "'";
704 int64_t poolid
= mon
->osdmon()->osdmap
.lookup_pg_pool_name(poolname
);
707 poolid
= strict_strtol(poolname
.c_str(), 10, &err
);
709 ss
<< "pool '" << poolname
<< "' does not exist";
711 } else if (poolid
< 0) {
712 ss
<< "invalid pool id '" << poolid
<< "'";
717 assert(poolid
>= 0); // Checked by parsing code above
719 if (fs
->mds_map
.get_first_data_pool() == poolid
) {
720 ss
<< "cannot remove default data pool";
726 fsmap
.modify_filesystem(fs
->fscid
,
727 [&r
, poolid
](std::shared_ptr
<Filesystem
> fs
)
729 r
= fs
->mds_map
.remove_data_pool(poolid
);
732 // It was already removed, succeed in silence
735 // We removed it, succeed
736 ss
<< "removed data pool " << poolid
<< " from fsmap";
739 // Unexpected error, bubble up
747 * For commands that refer to a particular filesystem,
748 * enable wrapping to implement the legacy version of
749 * the command (like "mds add_data_pool" vs "fs add_data_pool")
751 * The wrapped handler must expect a fs_name argument in
755 class LegacyHandler
: public T
757 std::string legacy_prefix
;
760 template <typename
... Args
>
761 LegacyHandler(const std::string
&new_prefix
, Args
&&... args
)
762 : T(std::forward
<Args
>(args
)...)
764 legacy_prefix
= new_prefix
;
767 std::string
const &get_prefix() override
{return legacy_prefix
;}
773 map
<string
, cmd_vartype
> &cmdmap
,
774 std::stringstream
&ss
) override
776 auto fs
= fsmap
.get_legacy_filesystem();
778 ss
<< "No filesystem configured";
781 std::map
<string
, cmd_vartype
> modified
= cmdmap
;
782 modified
["fs_name"] = fs
->mds_map
.get_fs_name();
783 return T::handle(mon
, fsmap
, op
, modified
, ss
);
788 * For commands with an alternative prefix
791 class AliasHandler
: public T
793 std::string alias_prefix
;
796 AliasHandler(const std::string
&new_prefix
)
799 alias_prefix
= new_prefix
;
802 std::string
const &get_prefix() override
{return alias_prefix
;}
808 map
<string
, cmd_vartype
> &cmdmap
,
809 std::stringstream
&ss
) override
811 return T::handle(mon
, fsmap
, op
, cmdmap
, ss
);
816 std::list
<std::shared_ptr
<FileSystemCommandHandler
> >
817 FileSystemCommandHandler::load(Paxos
*paxos
)
819 std::list
<std::shared_ptr
<FileSystemCommandHandler
> > handlers
;
821 handlers
.push_back(std::make_shared
<SetHandler
>());
822 handlers
.push_back(std::make_shared
<LegacyHandler
<SetHandler
> >("mds set"));
823 handlers
.push_back(std::make_shared
<FlagSetHandler
>());
824 handlers
.push_back(std::make_shared
<AddDataPoolHandler
>(paxos
));
825 handlers
.push_back(std::make_shared
<LegacyHandler
<AddDataPoolHandler
> >(
826 "mds add_data_pool", paxos
));
827 handlers
.push_back(std::make_shared
<RemoveDataPoolHandler
>());
828 handlers
.push_back(std::make_shared
<LegacyHandler
<RemoveDataPoolHandler
> >(
829 "mds remove_data_pool"));
830 handlers
.push_back(std::make_shared
<LegacyHandler
<RemoveDataPoolHandler
> >(
831 "mds rm_data_pool"));
832 handlers
.push_back(std::make_shared
<FsNewHandler
>(paxos
));
833 handlers
.push_back(std::make_shared
<RemoveFilesystemHandler
>());
834 handlers
.push_back(std::make_shared
<ResetFilesystemHandler
>());
836 handlers
.push_back(std::make_shared
<SetDefaultHandler
>());
837 handlers
.push_back(std::make_shared
<AliasHandler
<SetDefaultHandler
> >(
843 int FileSystemCommandHandler::parse_bool(
844 const std::string
&bool_str
,
848 assert(result
!= nullptr);
851 int64_t n
= strict_strtoll(bool_str
.c_str(), 10, &interr
);
853 if (bool_str
== "false" || bool_str
== "no"
854 || (interr
.length() == 0 && n
== 0)) {
857 } else if (bool_str
== "true" || bool_str
== "yes"
858 || (interr
.length() == 0 && n
== 1)) {
862 ss
<< "value must be false|no|0 or true|yes|1";
867 int FileSystemCommandHandler::_check_pool(
869 const int64_t pool_id
,
872 std::stringstream
*ss
) const
876 const pg_pool_t
*pool
= osd_map
.get_pg_pool(pool_id
);
878 *ss
<< "pool id '" << pool_id
<< "' does not exist";
882 const string
& pool_name
= osd_map
.get_pool_name(pool_id
);
884 if (pool
->is_erasure() && metadata
) {
885 *ss
<< "pool '" << pool_name
<< "' (id '" << pool_id
<< "')"
886 << " is an erasure-coded pool. Use of erasure-coded pools"
887 << " for CephFS metadata is not permitted";
889 } else if (pool
->is_erasure() && !pool
->allows_ecoverwrites()) {
890 // non-overwriteable EC pools are only acceptable with a cache tier overlay
891 if (!pool
->has_tiers() || !pool
->has_read_tier() || !pool
->has_write_tier()) {
892 *ss
<< "pool '" << pool_name
<< "' (id '" << pool_id
<< "')"
893 << " is an erasure-coded pool, with no overwrite support";
897 // That cache tier overlay must be writeback, not readonly (it's the
898 // write operations like modify+truncate we care about support for)
899 const pg_pool_t
*write_tier
= osd_map
.get_pg_pool(
901 assert(write_tier
!= NULL
); // OSDMonitor shouldn't allow DNE tier
902 if (write_tier
->cache_mode
== pg_pool_t::CACHEMODE_FORWARD
903 || write_tier
->cache_mode
== pg_pool_t::CACHEMODE_READONLY
) {
904 *ss
<< "EC pool '" << pool_name
<< "' has a write tier ("
905 << osd_map
.get_pool_name(pool
->write_tier
)
906 << ") that is configured "
907 "to forward writes. Use a cache mode such as 'writeback' for "
913 if (pool
->is_tier()) {
914 *ss
<< " pool '" << pool_name
<< "' (id '" << pool_id
915 << "') is already in use as a cache tier.";
919 if (!force
&& !pool
->application_metadata
.empty() &&
920 pool
->application_metadata
.count(
921 pg_pool_t::APPLICATION_NAME_CEPHFS
) == 0) {
922 *ss
<< " pool '" << pool_name
<< "' (id '" << pool_id
923 << "') has a non-CephFS application enabled.";
927 // Nothing special about this pool, so it is permissible