blob: f1e7ee798a0ce8cde225c2556b086b80e7635071 [file] [log] [blame]
Patrick Venturec0f499b2018-09-14 17:57:42 -07001#include "example/example.hpp"
2
Patrick Williams52509572023-05-10 07:51:18 -05003#include <phosphor-logging/log.hpp>
4
Patrick Venturec0f499b2018-09-14 17:57:42 -07005#include <algorithm>
Patrick Venturee01c0af2018-09-27 14:56:42 -07006#include <cstring>
Patrick Venture5100a382018-09-27 10:40:50 -07007#include <memory>
Patrick Venturec0f499b2018-09-14 17:57:42 -07008#include <string>
9#include <vector>
10
11namespace blobs
12{
13
Patrick Venture5100a382018-09-27 10:40:50 -070014using namespace phosphor::logging;
15
Patrick Venturec0f499b2018-09-14 17:57:42 -070016constexpr char ExampleBlobHandler::supportedPath[];
17
18ExampleBlob* ExampleBlobHandler::getSession(uint16_t id)
19{
20 auto search = sessions.find(id);
21 if (search == sessions.end())
22 {
23 return nullptr;
24 }
25 /* Not thread-safe, however, the blob handler deliberately assumes serial
26 * execution. */
27 return &search->second;
28}
29
30bool ExampleBlobHandler::canHandleBlob(const std::string& path)
31{
32 return (path == supportedPath);
33}
34
35std::vector<std::string> ExampleBlobHandler::getBlobIds()
36{
37 return {supportedPath};
38}
39
William A. Kennington III993f5412021-06-15 18:19:18 -070040bool ExampleBlobHandler::deleteBlob(const std::string&)
Patrick Venturec0f499b2018-09-14 17:57:42 -070041{
42 return false;
43}
44
William A. Kennington III993f5412021-06-15 18:19:18 -070045bool ExampleBlobHandler::stat(const std::string&, BlobMeta*)
Patrick Venturec0f499b2018-09-14 17:57:42 -070046{
47 return false;
48}
49
50bool ExampleBlobHandler::open(uint16_t session, uint16_t flags,
51 const std::string& path)
52{
53 if (!canHandleBlob(path))
54 {
55 return false;
56 }
57
58 auto findSess = sessions.find(session);
59 if (findSess != sessions.end())
60 {
61 /* This session is already active. */
62 return false;
63 }
64 sessions[session] = ExampleBlob(session, flags);
65 return true;
66}
67
68std::vector<uint8_t> ExampleBlobHandler::read(uint16_t session, uint32_t offset,
69 uint32_t requestedSize)
70{
71 ExampleBlob* sess = getSession(session);
72 if (!sess)
73 {
74 return std::vector<uint8_t>();
75 }
76
77 /* Is the offset beyond the array? */
78 if (offset >= sizeof(sess->buffer))
79 {
80 return std::vector<uint8_t>();
81 }
82
83 /* Determine how many bytes we can read from the offset.
84 * In this case, if they read beyond "size" we allow it.
85 */
86 uint32_t remain = sizeof(sess->buffer) - offset;
87 uint32_t numBytes = std::min(remain, requestedSize);
88 /* Copy the bytes! */
89 std::vector<uint8_t> result(numBytes);
90 std::memcpy(result.data(), &sess->buffer[offset], numBytes);
91 return result;
92}
93
94bool ExampleBlobHandler::write(uint16_t session, uint32_t offset,
95 const std::vector<uint8_t>& data)
96{
97 ExampleBlob* sess = getSession(session);
98 if (!sess)
99 {
100 return false;
101 }
102 /* Is the offset beyond the array? */
103 if (offset >= sizeof(sess->buffer))
104 {
105 return false;
106 }
107 /* Determine whether all their bytes will fit. */
108 uint32_t remain = sizeof(sess->buffer) - offset;
109 if (data.size() > remain)
110 {
111 return false;
112 }
113 sess->length =
114 std::max(offset + data.size(),
115 static_cast<std::vector<uint8_t>::size_type>(sess->length));
116 std::memcpy(&sess->buffer[offset], data.data(), data.size());
117 return true;
118}
119
William A. Kennington III993f5412021-06-15 18:19:18 -0700120bool ExampleBlobHandler::writeMeta(uint16_t, uint32_t,
121 const std::vector<uint8_t>&)
Patrick Venture5c4b17b2018-10-04 10:32:22 -0700122{
123 /* Not supported. */
124 return false;
125}
126
William A. Kennington III993f5412021-06-15 18:19:18 -0700127bool ExampleBlobHandler::commit(uint16_t session, const std::vector<uint8_t>&)
Patrick Venturec0f499b2018-09-14 17:57:42 -0700128{
129 ExampleBlob* sess = getSession(session);
130 if (!sess)
131 {
132 return false;
133 }
134
135 /* Do something with the staged data!. */
136
137 return false;
138}
139
140bool ExampleBlobHandler::close(uint16_t session)
141{
142 ExampleBlob* sess = getSession(session);
143 if (!sess)
144 {
145 return false;
146 }
147
148 sessions.erase(session);
149 return true;
150}
151
Patrick Venture8bc11772019-06-04 07:20:24 -0700152bool ExampleBlobHandler::stat(uint16_t session, BlobMeta* meta)
Patrick Venturec0f499b2018-09-14 17:57:42 -0700153{
154 ExampleBlob* sess = getSession(session);
155 if (!sess)
156 {
157 return false;
158 }
159 if (!meta)
160 {
161 return false;
162 }
163 meta->size = sess->length;
164 meta->blobState = sess->state;
165 return true;
166}
167
168bool ExampleBlobHandler::expire(uint16_t session)
169{
170 ExampleBlob* sess = getSession(session);
171 if (!sess)
172 {
173 return false;
174 }
175 /* TODO: implement session expiration behavior. */
176 return false;
177}
178
Patrick Venture5100a382018-09-27 10:40:50 -0700179void setupExampleHandler() __attribute__((constructor));
180
181void setupExampleHandler()
182{
Patrick Venture6c415c62018-11-14 14:01:36 -0800183 // You don't need to do anything in the constructor.
Patrick Venture5100a382018-09-27 10:40:50 -0700184}
185
Patrick Venturec0f499b2018-09-14 17:57:42 -0700186} // namespace blobs
Patrick Venture6c415c62018-11-14 14:01:36 -0800187
188/**
189 * This method is required by the blob manager.
190 *
191 * It is called to grab a handler for registering the blob handler instance.
192 */
193std::unique_ptr<blobs::GenericBlobInterface> createHandler()
194{
195 return std::make_unique<blobs::ExampleBlobHandler>();
196}