admin管理员组文章数量:1350369
I'm creating GPIO abstraction driver for stm32. There should be a class with register accesses and another class that holds a reference to it. In that way the class holding a reference has no idea about what is happening under the hood and doesn't care if i want to switch to a completely different mcu.
It is fine to make a class that keeps Port and Pin variables in Ram and inlines register calls, but it becomes problematic on low memory devices, such as f0.
The plan is to write a register accessing class as a template that takes Port and Pin as arguments. This way compiler will generate inline functions for each object created with a pair of <Port, Pin> and not keep them in Ram. Port and Pin are predefined Macros from CMSIS header.
Please ignore struct instead of class. Here is the simplified non template version:
struct Hal_Gpio
{
GPIO_TypeDef* port;
uint16_t pin;
Hal_Gpio(GPIO_TypeDef* port, uint16_t pin) : port(port), pin(pin)
{
}
bool read(void)
{
return port->IDR & pin;
}
};
struct Interface_Gpio
{
Hal_Gpio& pin;
Interface_Gpio(Hal_Gpio& pin) : pin(pin)
{
}
bool read(void)
{
return pin.read();
}
};
I've tried this template version. It works and uses 8 bytes less of Ram:
template<uint32_t port, uint32_t pin>
struct Hal_Gpio_template
{
bool read(void)
{
return reinterpret_cast<GPIO_TypeDef*>(port)->IDR & pin;
}
};
I've also tried this version, works.
template<class Port, Port port, class Pin, Pin pin>
struct Hal_Gpio_template
{
bool read(void)
{
return reinterpret_cast<GPIO_TypeDef*>(port)->IDR & pin;
}
};
I can not figure out a way to put this template class as a reference in a Interface_Gpio class.
Also objects instantiated from the template class cannot be put in the same array. This prevents having a loop that scans through n inputs.
I'm creating GPIO abstraction driver for stm32. There should be a class with register accesses and another class that holds a reference to it. In that way the class holding a reference has no idea about what is happening under the hood and doesn't care if i want to switch to a completely different mcu.
It is fine to make a class that keeps Port and Pin variables in Ram and inlines register calls, but it becomes problematic on low memory devices, such as f0.
The plan is to write a register accessing class as a template that takes Port and Pin as arguments. This way compiler will generate inline functions for each object created with a pair of <Port, Pin> and not keep them in Ram. Port and Pin are predefined Macros from CMSIS header.
Please ignore struct instead of class. Here is the simplified non template version:
struct Hal_Gpio
{
GPIO_TypeDef* port;
uint16_t pin;
Hal_Gpio(GPIO_TypeDef* port, uint16_t pin) : port(port), pin(pin)
{
}
bool read(void)
{
return port->IDR & pin;
}
};
struct Interface_Gpio
{
Hal_Gpio& pin;
Interface_Gpio(Hal_Gpio& pin) : pin(pin)
{
}
bool read(void)
{
return pin.read();
}
};
I've tried this template version. It works and uses 8 bytes less of Ram:
template<uint32_t port, uint32_t pin>
struct Hal_Gpio_template
{
bool read(void)
{
return reinterpret_cast<GPIO_TypeDef*>(port)->IDR & pin;
}
};
I've also tried this version, works.
template<class Port, Port port, class Pin, Pin pin>
struct Hal_Gpio_template
{
bool read(void)
{
return reinterpret_cast<GPIO_TypeDef*>(port)->IDR & pin;
}
};
I can not figure out a way to put this template class as a reference in a Interface_Gpio class.
Also objects instantiated from the template class cannot be put in the same array. This prevents having a loop that scans through n inputs.
Share Improve this question edited Apr 2 at 7:36 463035818_is_not_an_ai 124k11 gold badges102 silver badges214 bronze badges asked Apr 1 at 19:20 NikNik 112 bronze badges New contributor Nik is a new contributor to this site. Take care in asking for clarification, commenting, and answering. Check out our Code of Conduct. 5 |2 Answers
Reset to default 1Building on Botje's answer, why not just go with function pointers and explicitly instantiated function templates:
template<uint32_t port, uint32_t pin>
bool Hal_Gpio_Reader()
{
return reinterpret_cast<GPIO_TypeDef*>(port)->IDR & (pin);
}
struct Interface_Gpio
{
bool (*pin)();
Interface_Gpio(bool (*pin)()) : pin(pin)
{
}
bool read(void)
{
return pin();
}
};
Interface_Gpio my_gpio(&Hal_Gpio_Reader<1, 2>);
If you insist on using objects you probably must store some data:
In the way you've already tried you have to store port and pin.
You can use interface with abstract method and derive empty template class which holds everything in code generated from the template but you incur a cost of virtual method table pointer and a call.
== edit ==
Virtual method approach
class Hal_Gpio_Reader_Interface
{
public:
virtual bool operator()() = 0;
protected:
~Hal_Gpio_Reader_Interface() = default; // or make it public and virtual if there is a need to delete via base class pointer
};
template<uint32_t port, uint32_t pin>
struct Hal_Gpio_Reader : Hal_Gpio_Reader_Interface
{
bool operator()() final
{
return reinterpret_cast<GPIO_TypeDef*>(port)->IDR & pin;
}
};
struct Interface_Gpio
{
Hal_Gpio_Reader_Interface &pin; // this may have to be a pointer
Interface_Gpio(Hal_Gpio_Reader_Interface &pin) : pin(pin)
{
}
bool read(void)
{
return pin();
}
};
Hal_Gpio_Reader<1, 2> pin_1_2; // we need named object to take its reference
Interface_Gpio my_gpio(pin_1_2);
since you don't seem to mind trading a bit of code size for less RAM usage, you can abuse the fact that a no-capture lambda decays to a plain function pointer:
using Reader = bool(*)();
#define make_reader(port, pin) \
[](){ return reinterpret_cast<GPIO_TypeDef*>(port)->IDR & (pin); }
(it has to be a macro because the equivalent function requires the lambda to capture values)
You can now put a Reader
in your Interface_Gpio
and it will work.
Likewise:
std::array<Reader, 2> readers{ make_reader(1,2), make_reader(1,3) };
本文标签: C template class as a member in a normal classStack Overflow
版权声明:本文标题:C++ template class as a member in a normal class - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743876938a2554499.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
Hal_Gpio_template
is syntactically incorrect. Please copy-paste the correct code. – Ted Lyngmo Commented Apr 1 at 19:32