F´ Flight Software - C/C++ Documentation
A framework for building embedded system applications to NASA flight quality standards.
LinuxGpioDriver.cpp
Go to the documentation of this file.
1 // ======================================================================
2 // \title LinuxGpioDriverImpl.cpp
3 // \author tcanham
4 // \brief cpp file for LinuxGpioDriver component implementation class
5 //
6 // \copyright
7 // Copyright 2009-2015, by the California Institute of Technology.
8 // ALL RIGHTS RESERVED. United States Government Sponsorship
9 // acknowledged.
10 //
11 // ======================================================================
13 #include <Fw/FPrimeBasicTypes.hpp>
14 #include <Fw/Types/String.hpp>
15 #include <Fw/Types/StringUtils.hpp>
16 #include <Os/Posix/File.hpp>
17 
18 #include <linux/gpio.h>
19 #include <poll.h>
20 #include <sys/ioctl.h>
21 #include <unistd.h>
22 #include <cerrno>
23 #include <type_traits>
24 
25 namespace Drv {
26 
29  switch (errno_input) {
30  case 0:
31  status = Os::File::Status::OP_OK;
32  break;
33  case EBADF:
34  status = Os::File::Status::NOT_OPENED;
35  break;
36  case EINVAL:
37  status = Os::File::Status::INVALID_ARGUMENT;
38  break;
39  case ENODEV:
40  status = Os::File::Status::DOESNT_EXIST;
41  break;
42  case ENOMEM:
44  break;
45  case EPERM:
46  status = Os::File::Status::NO_PERMISSION;
47  break;
48  case ENXIO:
49  status = Os::File::Status::INVALID_MODE;
50  break;
51  // Cascades intended
52  case EFAULT:
53  case EWOULDBLOCK:
54  case EBUSY:
55  case EIO:
56  default:
58  break;
59  }
60  return status;
61 }
62 
64  Drv::GpioStatus status = Drv::GpioStatus::T::UNKNOWN_ERROR;
65  switch (errno_input) {
66  case EBADF:
67  status = Drv::GpioStatus::T::NOT_OPENED;
68  break;
69  case ENXIO:
71  break;
72  // Cascades intended
73  case EFAULT:
74  case EINVAL:
75  case EWOULDBLOCK:
76  case EBUSY:
77  case EIO:
78  default:
79  status = Drv::GpioStatus::T::UNKNOWN_ERROR;
80  break;
81  }
82  return status;
83 }
84 
86  U32 flags = 0;
87  switch (configuration) {
89  flags = GPIOHANDLE_REQUEST_OUTPUT;
90  break;
91  // Cascade intended
96  flags = GPIOHANDLE_REQUEST_INPUT;
97  break;
98  default:
99  FW_ASSERT(0, static_cast<FwAssertArgType>(configuration));
100  break;
101  }
102  return flags;
103 }
104 
106  U32 flags = 0;
107  switch (configuration) {
109  flags = GPIOEVENT_REQUEST_RISING_EDGE;
110  break;
112  flags = GPIOEVENT_REQUEST_FALLING_EDGE;
113  break;
115  flags = GPIOEVENT_REQUEST_RISING_EDGE | GPIOEVENT_REQUEST_FALLING_EDGE;
116  break;
117  default:
118  FW_ASSERT(0, static_cast<FwAssertArgType>(configuration));
119  break;
120  }
121  return flags;
122 }
123 
125  (void)::close(this->m_fd);
126 }
127 
128 // ----------------------------------------------------------------------
129 // Handler implementations for user-defined typed input ports
130 // ----------------------------------------------------------------------
131 
132 Os::File::Status LinuxGpioDriver ::setupLineHandle(const int chip_descriptor,
133  const U32 gpio,
134  const GpioConfiguration& configuration,
135  const Fw::Logic& default_state,
136  int& fd) {
138  // Set up the GPIO request
139  struct gpiohandle_request request;
140  (void)::memset(&request, 0, sizeof request);
141  request.lineoffsets[0] = gpio;
142  Fw::StringUtils::string_copy(request.consumer_label, FW_OPTIONAL_NAME(this->getObjName()),
143  static_cast<FwSizeType>(sizeof request.consumer_label));
144  request.default_values[0] = (default_state == Fw::Logic::HIGH) ? 1 : 0;
145  request.fd = -1;
146  request.lines = 1;
147  request.flags = configuration_to_handler_flags(configuration);
148 
149  errno = 0;
150  int return_value = ioctl(chip_descriptor, GPIO_GET_LINEHANDLE_IOCTL, &request);
151  fd = request.fd;
152  if (return_value != 0) {
153  status = errno_to_file_status(errno);
154  fd = -1;
155  }
156  return status;
157 }
158 
159 Os::File::Status LinuxGpioDriver ::setupLineEvent(const int chip_descriptor,
160  const U32 gpio,
161  const GpioConfiguration& configuration,
162  int& fd) {
164  // Set up the GPIO request
165  struct gpioevent_request event;
166  (void)::memset(&event, 0, sizeof event);
167  event.lineoffset = gpio;
168  Fw::StringUtils::string_copy(event.consumer_label, FW_OPTIONAL_NAME(this->getObjName()),
169  static_cast<FwSizeType>(sizeof event.consumer_label));
170  event.fd = -1;
171  event.handleflags = configuration_to_handler_flags(configuration);
172  event.eventflags = configuration_to_event_flags(configuration);
173  errno = 0;
174  int return_value = ioctl(chip_descriptor, GPIO_GET_LINEEVENT_IOCTL, &event);
175  fd = event.fd;
176  if (return_value != 0) {
177  status = errno_to_file_status(errno);
178  fd = -1;
179  }
180  return status;
181 }
182 
184  const U32 gpio,
185  const GpioConfiguration& configuration,
186  const Fw::Logic& default_state) {
188  Os::File chip_file;
189  FW_ASSERT(device != nullptr);
190  FW_ASSERT(configuration < MAX_GPIO_CONFIGURATION and configuration >= 0,
191  static_cast<FwAssertArgType>(configuration));
192 
193  // Open chip file and check for success
194  status = chip_file.open(device, Os::File::Mode::OPEN_WRITE);
195  if (status != Os::File::OP_OK) {
196  this->log_WARNING_HI_OpenChipError(Fw::String(device), Os::FileStatus(static_cast<Os::FileStatus::T>(status)));
197  return status;
198  }
199  // Read chip information and check for correctness
200  int chip_descriptor = reinterpret_cast<Os::Posix::File::PosixFileHandle*>(chip_file.getHandle())->m_file_descriptor;
201  struct gpiochip_info chip_info;
202  (void)::memset(&chip_info, 0, sizeof chip_info);
203  int return_value = ioctl(chip_descriptor, GPIO_GET_CHIPINFO_IOCTL, &chip_info);
204  if (return_value != 0) {
205  status = errno_to_file_status(errno);
206  this->log_WARNING_HI_OpenChipError(Fw::String(device), Os::FileStatus(static_cast<Os::FileStatus::T>(status)));
207  return status;
208  }
209  // Check if the GPIO line exists
210  if (gpio >= chip_info.lines) {
211  this->log_WARNING_HI_OpenPinError(Fw::String(device), gpio, Fw::String("Does Not Exist"),
212  Os::FileStatus(static_cast<Os::FileStatus::T>(status)));
213  return status;
214  }
215  Fw::String pin_message("Unknown");
216  struct gpioline_info pin_info;
217  (void)::memset(&pin_info, 0, sizeof pin_info);
218  pin_info.line_offset = gpio;
219  return_value = ioctl(chip_descriptor, GPIO_GET_LINEINFO_IOCTL, &pin_info);
220  if (return_value == 0) {
221  const bool has_consumer = pin_info.consumer[0] != '\0';
222  pin_message.format("%s%s%s", pin_info.name, has_consumer ? " with current consumer " : "",
223  has_consumer ? pin_info.consumer : "");
224  }
225 
226  // Set up pin and set file descriptor for it
227  int pin_fd = -1;
228  switch (configuration) {
229  // Cascade intended
230  case GPIO_OUTPUT:
231  case GPIO_INPUT:
232  status = this->setupLineHandle(chip_descriptor, gpio, configuration, default_state, pin_fd);
233  break;
234  // Cascade intended
238  status = this->setupLineEvent(chip_descriptor, gpio, configuration, pin_fd);
239  break;
240  default:
241  FW_ASSERT(0);
242  break;
243  }
244  // Final status check
245  if (status != Os::File::Status::OP_OK) {
246  this->log_WARNING_HI_OpenPinError(Fw::String(device), gpio, pin_message,
247  Os::FileStatus(static_cast<Os::FileStatus::T>(status)));
248  } else {
249  this->log_DIAGNOSTIC_OpenChip(Fw::String(chip_info.name), Fw::String(chip_info.label), gpio, pin_message);
250  this->m_fd = pin_fd;
251  this->m_configuration = configuration;
252  }
253  return status;
254 }
255 
256 Drv::GpioStatus LinuxGpioDriver ::gpioRead_handler(const FwIndexType portNum, Fw::Logic& state) {
258  if (this->m_configuration == GpioConfiguration::GPIO_INPUT) {
259  struct gpiohandle_data values;
260  (void)::memset(&values, 0, sizeof values);
261  int return_value = ioctl(this->m_fd, GPIOHANDLE_GET_LINE_VALUES_IOCTL, &values);
262  if (return_value != 0) {
263  status = errno_to_gpio_status(errno);
264  } else {
265  state = values.values[0] ? Fw::Logic::HIGH : Fw::Logic::LOW;
266  status = Drv::GpioStatus::OP_OK;
267  }
268  }
269  return status;
270 }
271 
272 Drv::GpioStatus LinuxGpioDriver ::gpioWrite_handler(const FwIndexType portNum, const Fw::Logic& state) {
274  if (this->m_configuration == GpioConfiguration::GPIO_OUTPUT) {
275  struct gpiohandle_data values;
276  (void)::memset(&values, 0, sizeof values);
277  values.values[0] = (state == Fw::Logic::HIGH) ? 1 : 0;
278  int return_value = ioctl(this->m_fd, GPIOHANDLE_SET_LINE_VALUES_IOCTL, &values);
279  if (return_value != 0) {
280  status = errno_to_gpio_status(errno);
281  } else {
282  status = Drv::GpioStatus::OP_OK;
283  }
284  }
285  return status;
286 }
287 
288 void LinuxGpioDriver ::pollLoop() {
289  // Ensure size of FwSizeType is large enough to fit the necessary ranges
290  // NOTE: casts to unsigned types for int and ssize_t are made to avoid sign-compare warning;
291  // in both cases the cast is safe because max() returns nonnegative value.
292  static_assert(GPIO_POLL_TIMEOUT < static_cast<unsigned int>(std::numeric_limits<int>::max()),
293  "Poll timeout would overflow");
294  static_assert(sizeof(struct gpioevent_data) < std::numeric_limits<FwSizeType>::max(), "FwSizeType too small");
295  using unsigned_ssize_t = std::make_unsigned<ssize_t>::type;
296  static_assert(
297  static_cast<unsigned_ssize_t>(std::numeric_limits<ssize_t>::max()) <= std::numeric_limits<FwSizeType>::max(),
298  "FwSizeType too small");
299  // Setup poll information
300  pollfd file_descriptors[1];
301  // Loop forever
302  while (this->getRunning()) {
303  // Setup polling
304  (void)::memset(file_descriptors, 0, sizeof file_descriptors);
305  file_descriptors[0].fd = this->m_fd;
306  file_descriptors[0].events = POLLIN; // Ask for read data available
307  // Poll for fd bing ready
308  int status = ::poll(file_descriptors, 1, static_cast<int>(GPIO_POLL_TIMEOUT));
309  // Check for some file descriptor to be ready
310  if (status > 0) {
311  struct gpioevent_data event_data;
312  FwSizeType read_bytes = static_cast<FwSizeType>(::read(this->m_fd, &event_data, sizeof event_data));
313  if (read_bytes == sizeof event_data) {
314  Os::RawTime timestamp;
315  timestamp.now();
316  this->gpioInterrupt_out(0, timestamp);
317  }
318  // A read error occurred
319  else {
320  this->log_WARNING_HI_InterruptReadError(static_cast<U32>(sizeof event_data),
321  static_cast<U32>(read_bytes));
322  }
323  }
324  // An error of some kind occurred
325  else if (status < 0) {
326  this->log_WARNING_HI_PollingError(static_cast<I32>(errno));
327  }
328  }
329 }
330 
331 } // end namespace Drv
Operation succeeded.
Definition: Os.hpp:26
A catch-all for other errors. Have to look in implementation-specific code.
PlatformSizeType FwSizeType
Input GPIO pin triggers interrupt port on falling edge.
void log_WARNING_HI_InterruptReadError(U32 expected, U32 got) const
Log event InterruptReadError.
Status now() override
Get the current time.
Definition: RawTime.cpp:36
#define FW_OPTIONAL_NAME(name)
Definition: FpConfig.h:47
void log_WARNING_HI_OpenPinError(const Fw::StringBase &chip, U32 pin, const Fw::StringBase &pinMessage, Os::FileStatus status) const
Log event OpenPinError.
Os::FileInterface::Status open(const char *path, Mode mode)
open file with supplied path and mode
Operation succeeded.
Operation not permitted with current configuration.
void log_WARNING_HI_OpenChipError(const Fw::StringBase &chip, Os::FileStatus status) const
Log event OpenChipError.
char * string_copy(char *destination, const char *source, FwSizeType num)
copy string with null-termination guaranteed
Definition: StringUtils.cpp:7
void log_DIAGNOSTIC_OpenChip(const Fw::StringBase &chip, const Fw::StringBase &chipLabel, U32 pin, const Fw::StringBase &pinMessage) const
Log event OpenChip.
FileHandle * getHandle() override
returns the raw file handle
Definition: File.cpp:206
void gpioInterrupt_out(FwIndexType portNum, Os::RawTime &cycleStart)
Invoke output port gpioInterrupt.
GpioConfiguration
configure the GPIO pin
Input GPIO pin triggers interrupt port on both edges.
FormatStatus format(const CHAR *formatString,...)
write formatted string to buffer
Definition: StringBase.cpp:55
Logic low state.
Definition: LogicEnumAc.hpp:33
Drv::GpioStatus errno_to_gpio_status(int errno_input)
No space left on the device for writing.
Os::File::Status open(const char *device, const U32 gpio, const GpioConfiguration &configuration, const Fw::Logic &default_state=Fw::Logic::LOW)
open a GPIO pin for use in the system
Operation was successful.
Definition: File.hpp:40
PlatformIndexType FwIndexType
void log_WARNING_HI_PollingError(I32 error_number) const
Log event PollingError.
Output GPIO pin for direct writing.
Logic states.
Definition: LogicEnumAc.hpp:17
Logic high state.
Definition: LogicEnumAc.hpp:35
U32 configuration_to_event_flags(Drv::LinuxGpioDriver::GpioConfiguration configuration)
Input GPIO pin for direct reading.
Input GPIO pin triggers interrupt port on rising edge.
U32 configuration_to_handler_flags(Drv::LinuxGpioDriver::GpioConfiguration configuration)
FPP shadow-enum representing Os::File::Status.
Os::File::Status errno_to_file_status(int errno_input)
static constexpr FwSizeType GPIO_POLL_TIMEOUT
#define FW_ASSERT(...)
Definition: Assert.hpp:14