| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 1 | #include "watch.hpp" | 
|  | 2 |  | 
| Nan Zhou | 014be0b | 2021-12-28 18:00:14 -0800 | [diff] [blame] | 3 | #include <sys/epoll.h> | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 4 | #include <sys/inotify.h> | 
|  | 5 | #include <unistd.h> | 
|  | 6 |  | 
| Patrick Williams | 223e460 | 2023-05-10 07:51:11 -0500 | [diff] [blame] | 7 | #include <phosphor-logging/elog-errors.hpp> | 
|  | 8 | #include <phosphor-logging/elog.hpp> | 
| Ravi Teja | f264627 | 2023-09-30 13:00:55 -0500 | [diff] [blame] | 9 | #include <phosphor-logging/lg2.hpp> | 
| Patrick Williams | 223e460 | 2023-05-10 07:51:11 -0500 | [diff] [blame] | 10 | #include <sdeventplus/source/io.hpp> | 
|  | 11 | #include <xyz/openbmc_project/Common/error.hpp> | 
|  | 12 |  | 
| Nan Zhou | 014be0b | 2021-12-28 18:00:14 -0800 | [diff] [blame] | 13 | #include <array> | 
|  | 14 | #include <cerrno> | 
|  | 15 | #include <climits> | 
|  | 16 | #include <cstdint> | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 17 | #include <cstring> | 
|  | 18 | #include <filesystem> | 
| Nan Zhou | e1289ad | 2021-12-28 11:02:56 -0800 | [diff] [blame] | 19 |  | 
|  | 20 | namespace phosphor::certs | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 21 | { | 
| Nan Zhou | cf06ccd | 2021-12-28 16:25:45 -0800 | [diff] [blame] | 22 |  | 
|  | 23 | using ::phosphor::logging::elog; | 
| Nan Zhou | cf06ccd | 2021-12-28 16:25:45 -0800 | [diff] [blame] | 24 | using ::sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure; | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 25 | namespace fs = std::filesystem; | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 26 |  | 
|  | 27 | Watch::Watch(sdeventplus::Event& event, std::string& certFile, Callback cb) : | 
| Nan Zhou | 014be0b | 2021-12-28 18:00:14 -0800 | [diff] [blame] | 28 | event(event), callback(std::move(cb)) | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 29 | { | 
|  | 30 | // get parent directory of certificate file to watch | 
| Nan Zhou | cf06ccd | 2021-12-28 16:25:45 -0800 | [diff] [blame] | 31 | fs::path path = fs::path(certFile).parent_path(); | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 32 | try | 
|  | 33 | { | 
|  | 34 | if (!fs::exists(path)) | 
|  | 35 | { | 
|  | 36 | fs::create_directories(path); | 
|  | 37 | } | 
|  | 38 | } | 
| Patrick Williams | 7195799 | 2021-10-06 14:42:52 -0500 | [diff] [blame] | 39 | catch (const fs::filesystem_error& e) | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 40 | { | 
| Ravi Teja | f264627 | 2023-09-30 13:00:55 -0500 | [diff] [blame] | 41 | lg2::error( | 
|  | 42 | "Failed to create directory, ERR:{ERR}, DIRECTORY:{DIRECTORY}", | 
|  | 43 | "ERR", e, "DIRECTORY", path); | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 44 | elog<InternalFailure>(); | 
|  | 45 | } | 
|  | 46 | watchDir = path; | 
|  | 47 | watchFile = fs::path(certFile).filename(); | 
|  | 48 | startWatch(); | 
|  | 49 | } | 
|  | 50 |  | 
|  | 51 | Watch::~Watch() | 
|  | 52 | { | 
|  | 53 | stopWatch(); | 
|  | 54 | } | 
|  | 55 |  | 
|  | 56 | void Watch::startWatch() | 
|  | 57 | { | 
|  | 58 | // stop any existing watch | 
|  | 59 | stopWatch(); | 
|  | 60 |  | 
|  | 61 | fd = inotify_init1(IN_NONBLOCK); | 
|  | 62 | if (-1 == fd) | 
|  | 63 | { | 
| Ravi Teja | f264627 | 2023-09-30 13:00:55 -0500 | [diff] [blame] | 64 | lg2::error("inotify_init1 failed: {ERR}", "ERR", std::strerror(errno)); | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 65 | elog<InternalFailure>(); | 
|  | 66 | } | 
|  | 67 | wd = inotify_add_watch(fd, watchDir.c_str(), IN_CLOSE_WRITE); | 
|  | 68 | if (-1 == wd) | 
|  | 69 | { | 
|  | 70 | close(fd); | 
| Ravi Teja | f264627 | 2023-09-30 13:00:55 -0500 | [diff] [blame] | 71 | lg2::error("inotify_add_watch failed, ERR:{ERR}, WATCH:{WATCH}", "ERR", | 
|  | 72 | std::strerror(errno), "WATCH", watchDir); | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 73 | elog<InternalFailure>(); | 
|  | 74 | } | 
|  | 75 |  | 
|  | 76 | ioPtr = std::make_unique<sdeventplus::source::IO>( | 
| Patrick Williams | 7e2797e | 2021-12-03 13:37:07 -0600 | [diff] [blame] | 77 | event, fd, EPOLLIN, [this](sdeventplus::source::IO&, int fd, uint32_t) { | 
| Patrick Williams | d96b81c | 2023-10-20 11:19:39 -0500 | [diff] [blame] | 78 | constexpr int size = sizeof(struct inotify_event) + NAME_MAX + 1; | 
|  | 79 | std::array<char, size> buffer{}; | 
|  | 80 | int length = read(fd, buffer.data(), buffer.size()); | 
|  | 81 | if (length >= static_cast<int>(sizeof(struct inotify_event))) | 
|  | 82 | { | 
|  | 83 | struct inotify_event* notifyEvent = | 
|  | 84 | reinterpret_cast<struct inotify_event*>(&buffer[0]); | 
|  | 85 | if (notifyEvent->len) | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 86 | { | 
| Patrick Williams | d96b81c | 2023-10-20 11:19:39 -0500 | [diff] [blame] | 87 | if (watchFile == notifyEvent->name) | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 88 | { | 
| Patrick Williams | d96b81c | 2023-10-20 11:19:39 -0500 | [diff] [blame] | 89 | callback(); | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 90 | } | 
|  | 91 | } | 
| Patrick Williams | d96b81c | 2023-10-20 11:19:39 -0500 | [diff] [blame] | 92 | } | 
|  | 93 | else | 
|  | 94 | { | 
|  | 95 | lg2::error("Failed to read inotify event"); | 
|  | 96 | } | 
|  | 97 | }); | 
| Marri Devender Rao | ffad1ef | 2019-06-03 04:54:12 -0500 | [diff] [blame] | 98 | } | 
|  | 99 |  | 
|  | 100 | void Watch::stopWatch() | 
|  | 101 | { | 
|  | 102 | if (-1 != fd) | 
|  | 103 | { | 
|  | 104 | if (-1 != wd) | 
|  | 105 | { | 
|  | 106 | inotify_rm_watch(fd, wd); | 
|  | 107 | } | 
|  | 108 | close(fd); | 
|  | 109 | } | 
|  | 110 | if (ioPtr) | 
|  | 111 | { | 
|  | 112 | ioPtr.reset(); | 
|  | 113 | } | 
|  | 114 | } | 
|  | 115 |  | 
| Nan Zhou | e1289ad | 2021-12-28 11:02:56 -0800 | [diff] [blame] | 116 | } // namespace phosphor::certs |