OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
ossia-max.hpp
1#pragma once
2#define OSSIA_MAX_AUTOREGISTER 1
3#include "ext.h"
4#include "ext_obex.h"
5#include "jpatcher_api.h"
6
7#include <ossia/detail/config.hpp>
8#undef error
9#undef post
10
11#include "ZeroconfMinuitListener.hpp"
12#include "ZeroconfOscqueryListener.hpp"
13#include "assert.hpp"
14#include "attribute.hpp"
15#include "client.hpp"
16#include "ocue.hpp"
17#include "device.hpp"
18#include "explorer.hpp"
19#include "fuzzysearch.hpp"
20#include "logger.hpp"
21#include "model.hpp"
22#include "monitor.hpp"
23#include "ossia_object.hpp"
24#include "parameter.hpp"
25#include "remote.hpp"
26#include "router.hpp"
27#include "search.hpp"
28#include "view.hpp"
29
30#include <ossia/detail/hash_map.hpp>
31#include <ossia/detail/safe_vec.hpp>
32#include <ossia/network/common/websocket_log_sink.hpp>
33
34#include <ossia-max/src/object_base.hpp>
35
36#include <ossia-max_export.h>
37
38extern "C" {
39OSSIA_MAX_EXPORT void ossia_router_setup();
40OSSIA_MAX_EXPORT void ossia_attribute_setup();
41OSSIA_MAX_EXPORT void ossia_client_setup();
42OSSIA_MAX_EXPORT void ossia_device_setup();
43OSSIA_MAX_EXPORT void ossia_logger_setup();
44OSSIA_MAX_EXPORT void ossia_model_setup();
45OSSIA_MAX_EXPORT void ossia_parameter_setup();
46OSSIA_MAX_EXPORT void ossia_remote_setup();
47OSSIA_MAX_EXPORT void ossia_view_setup();
48OSSIA_MAX_EXPORT void ossia_ossia_setup();
49OSSIA_MAX_EXPORT void ossia_explorer_setup();
50OSSIA_MAX_EXPORT void ossia_search_setup();
51OSSIA_MAX_EXPORT void ossia_monitor_setup();
52OSSIA_MAX_EXPORT void ossia_fuzzysearch_setup();
53OSSIA_MAX_EXPORT void ossia_assert_setup();
54OSSIA_MAX_EXPORT void ossia_cue_setup();
55OSSIA_MAX_EXPORT void ossia_equals_setup();
56}
57
58namespace ossia
59{
60namespace max_binding
61{
62struct ocue;
63#pragma mark -
64#pragma mark Library
65
66struct max_msp_log_sink final : public spdlog::sinks::sink
67{
68 void log(const spdlog::details::log_msg& msg) override
69 {
70 std::string s(msg.payload.data(), msg.payload.size());
71 switch(msg.level)
72 {
73 case spdlog::level::warn:
74 case spdlog::level::err: {
75 error("%s", s.c_str());
76 break;
77 }
78
79 default:
80 post("%s", s.c_str());
81 break;
82 }
83 }
84
85 void flush() override { }
86
87 void set_pattern(const std::string& pattern) override { }
88 void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) override { }
89};
90
91struct patcher_descriptor
92{
93 ossia::safe_set<parameter*> parameters{};
94 ossia::safe_set<remote*> remotes{};
95 ossia::safe_set<attribute*> attributes{};
96 ossia::safe_set<ocue*> cues{};
97 model* model{};
98 view* view{};
99 device* device{};
100 client* client{};
101
102 t_object* parent_patcher{};
103 ossia::safe_set<t_object*> subpatchers{};
104
105 bool loadbanged{}; // true if patcher have been loadbanged already
106
107 long poly_index{
108 -1}; // 0 when not in a poly~, instance index otherwise, -1 uninitialized
109
110 bool empty() const
111 {
112 return parameters.empty() && remotes.empty() && attributes.empty() && cues.empty()
113 && model != nullptr && view != nullptr && device != nullptr
114 && client != nullptr;
115 }
116
117 auto size() const
118 {
119 return parameters.size() + remotes.size() + attributes.size() + cues.size()
120 + (model ? 1 : 0) + (view ? 1 : 0) + (device ? 1 : 0) + (client ? 1 : 0);
121 }
122
123 bool has_no_master_node()
124 {
125 return model == nullptr && view == nullptr && device == nullptr && client == nullptr;
126 }
127};
128
129struct configuration
130{
131 bool defer_by_default = true;
132 bool autoregister = true;
133};
134
135template <typename T>
136t_class* ossia_cue_class{};
137
138class ossia_max
139{
140public:
141 static ossia_max& instance();
142 static const std::shared_ptr<ossia::net::generic_device>& get_default_device()
143 {
144 return instance().m_device;
145 }
146
147 // static void register_nodes(ossia_max* x);
148 static void discover_network_devices(ossia_max* x);
149
150 template <typename T>
151 t_class* get_class()
152 {
153 if constexpr(std::is_same<T, parameter>::value)
154 return ossia_parameter_class;
155 else if constexpr(std::is_same<T, remote>::value)
156 return ossia_remote_class;
157 else if constexpr(std::is_same<T, model>::value)
158 return ossia_model_class;
159 else if constexpr(std::is_same<T, view>::value)
160 return ossia_view_class;
161 else if constexpr(std::is_same<T, device>::value)
162 return ossia_device_class;
163 else if constexpr(std::is_same<T, client>::value)
164 return ossia_client_class;
165 else if constexpr(std::is_same<T, attribute>::value)
166 return ossia_attribute_class;
167 else if constexpr(std::is_same<T, ossia_object>::value)
168 return ossia_ossia_class;
169 else if constexpr(std::is_same<T, ossia::max_binding::logger>::value)
170 return ossia_logger_class;
171 else if constexpr(std::is_same<T, ossia::max_binding::explorer>::value)
172 return ossia_explorer_class;
173 else if constexpr(std::is_same<T, ossia::max_binding::monitor>::value)
174 return ossia_monitor_class;
175 else if constexpr(std::is_same<T, ossia::max_binding::search>::value)
176 return ossia_search_class;
177 else if constexpr(std::is_same<T, ossia::max_binding::router>::value)
178 return ossia_router_class;
179 else if constexpr(std::is_same<T, ossia::max_binding::fuzzysearch>::value)
180 return ossia_fuzzysearch_class;
181 else if constexpr(std::is_same<T, ossia::max_binding::oassert>::value)
182 return ossia_assert_class;
183 else if constexpr(requires { T::max_class; })
184 return T::max_class;
185 return nullptr;
186 }
187
188 void set_log_level(t_symbol* log_sym)
189 {
190 std::vector<spdlog::string_view_t> vec SPDLOG_LEVEL_NAMES;
191 auto it = std::find(vec.begin(), vec.end(), log_sym->s_name);
192 if(it != vec.end())
193 {
194 int level = it - vec.begin();
195 m_log_sink.get()->set_level(static_cast<spdlog::level::level_enum>(level));
196 }
197 else
198 {
199 error("Unknown log level : %s", log_sym->s_name);
200 }
201 }
202
203 t_class* ossia_router_class{};
204 t_class* ossia_client_class{};
205 t_class* ossia_attribute_class{};
206 t_class* ossia_device_class{};
207 t_class* ossia_explorer_class{};
208 t_class* ossia_monitor_class{};
209 t_class* ossia_search_class{};
210 t_class* ossia_fuzzysearch_class{};
211 t_class* ossia_logger_class{};
212 t_class* ossia_model_class{};
213 t_class* ossia_parameter_class{};
214 t_class* ossia_remote_class{};
215 t_class* ossia_view_class{};
216 t_class* ossia_ossia_class{};
217 t_class* ossia_assert_class{};
218 t_class* ossia_equals_class{};
219 static t_class* ossia_patcher_listener_class;
220
221 // keep list of all objects
222 // TODO is it still needed ?
223 ossia::safe_vector<remote*> remotes;
224 ossia::safe_vector<view*> views;
225 ossia::safe_vector<device*> devices;
226 ossia::safe_vector<client*> clients;
227 ossia::safe_vector<logger*> loggers;
228
229 static ossia::hash_map<ossia::net::node_base*, ossia::safe_set<matcher*>>
230 s_node_matchers_map;
231 static std::recursive_mutex s_node_matchers_mut;
232
233 // TODO is this still needed ?
234 bool registering_nodes = false;
235
236 std::map<t_object*, patcher_descriptor> patchers;
237
238 static patcher_descriptor& get_patcher_descriptor(t_object* patcher);
239 static void remove_patcher_descriptor(t_object* patcher);
240
241 void* m_reg_clock{};
242 static void* s_browse_clock;
243
244 static ZeroconfOscqueryListener s_zeroconf_oscq_listener;
245 static ZeroconfMinuitListener s_zeroconf_minuit_listener;
246
247 configuration config;
248
249private:
250 ossia_max();
251 ~ossia_max();
252 ossia_max(const ossia_max&) = delete;
253 ossia_max(ossia_max&&) = delete;
254
255 ossia::net::local_protocol* m_localProtocol{};
256 std::shared_ptr<ossia::net::generic_device> m_device;
257 string_map<std::shared_ptr<ossia::websocket_threaded_connection>> m_connections;
258 std::shared_ptr<max_msp_log_sink> m_log_sink;
259
260 t_object* m_patcher_listener;
261};
262
263template <typename T>
264T* make_ossia()
265{
266 auto obj = object_alloc(ossia_max::instance().get_class<T>());
267 if(obj)
268 {
269 t_object tmp;
270 memcpy(&tmp, obj, sizeof(t_object));
271 auto x = new(obj) T{};
272 memcpy(x, &tmp, sizeof(t_object));
273
274 return x;
275 }
276 return nullptr;
277}
278
279} // max namespace
280} // ossia namespace
A protocol used to expose a local application's data through multiple other protocols.
Definition local.hpp:22
Definition git_info.h:7