obs-internal.h 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323
  1. /******************************************************************************
  2. Copyright (C) 2023 by Lain Bailey <[email protected]>
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. ******************************************************************************/
  14. #pragma once
  15. #include "util/c99defs.h"
  16. #include "util/darray.h"
  17. #include "util/circlebuf.h"
  18. #include "util/dstr.h"
  19. #include "util/threading.h"
  20. #include "util/platform.h"
  21. #include "util/profiler.h"
  22. #include "util/task.h"
  23. #include "util/uthash.h"
  24. #include "callback/signal.h"
  25. #include "callback/proc.h"
  26. #include "graphics/graphics.h"
  27. #include "graphics/matrix4.h"
  28. #include "media-io/audio-resampler.h"
  29. #include "media-io/video-io.h"
  30. #include "media-io/audio-io.h"
  31. #include "obs.h"
  32. #include <obsversion.h>
  33. #include <caption/caption.h>
  34. /* Custom helpers for the UUID hash table */
  35. #define HASH_FIND_UUID(head, uuid, out) \
  36. HASH_FIND(hh_uuid, head, uuid, UUID_STR_LENGTH, out)
  37. #define HASH_ADD_UUID(head, uuid_field, add) \
  38. HASH_ADD(hh_uuid, head, uuid_field[0], UUID_STR_LENGTH, add)
  39. #define NUM_TEXTURES 2
  40. #define NUM_CHANNELS 3
  41. #define MICROSECOND_DEN 1000000
  42. #define NUM_ENCODE_TEXTURES 10
  43. #define NUM_ENCODE_TEXTURE_FRAMES_TO_WAIT 1
  44. static inline int64_t packet_dts_usec(struct encoder_packet *packet)
  45. {
  46. return packet->dts * MICROSECOND_DEN / packet->timebase_den;
  47. }
  48. struct tick_callback {
  49. void (*tick)(void *param, float seconds);
  50. void *param;
  51. };
  52. struct draw_callback {
  53. void (*draw)(void *param, uint32_t cx, uint32_t cy);
  54. void *param;
  55. };
  56. struct rendered_callback {
  57. void (*rendered)(void *param);
  58. void *param;
  59. };
  60. /* ------------------------------------------------------------------------- */
  61. /* validity checks */
  62. static inline bool obs_object_valid(const void *obj, const char *f,
  63. const char *t)
  64. {
  65. if (!obj) {
  66. blog(LOG_DEBUG, "%s: Null '%s' parameter", f, t);
  67. return false;
  68. }
  69. return true;
  70. }
  71. #define obs_ptr_valid(ptr, func) obs_object_valid(ptr, func, #ptr)
  72. #define obs_source_valid obs_ptr_valid
  73. #define obs_output_valid obs_ptr_valid
  74. #define obs_encoder_valid obs_ptr_valid
  75. #define obs_service_valid obs_ptr_valid
  76. /* ------------------------------------------------------------------------- */
  77. /* modules */
  78. struct obs_module {
  79. char *mod_name;
  80. const char *file;
  81. char *bin_path;
  82. char *data_path;
  83. void *module;
  84. bool loaded;
  85. bool (*load)(void);
  86. void (*unload)(void);
  87. void (*post_load)(void);
  88. void (*set_locale)(const char *locale);
  89. bool (*get_string)(const char *lookup_string,
  90. const char **translated_string);
  91. void (*free_locale)(void);
  92. uint32_t (*ver)(void);
  93. void (*set_pointer)(obs_module_t *module);
  94. const char *(*name)(void);
  95. const char *(*description)(void);
  96. const char *(*author)(void);
  97. struct obs_module *next;
  98. };
  99. extern void free_module(struct obs_module *mod);
  100. struct obs_module_path {
  101. char *bin;
  102. char *data;
  103. };
  104. static inline void free_module_path(struct obs_module_path *omp)
  105. {
  106. if (omp) {
  107. bfree(omp->bin);
  108. bfree(omp->data);
  109. }
  110. }
  111. static inline bool check_path(const char *data, const char *path,
  112. struct dstr *output)
  113. {
  114. dstr_copy(output, path);
  115. dstr_cat(output, data);
  116. return os_file_exists(output->array);
  117. }
  118. /* ------------------------------------------------------------------------- */
  119. /* hotkeys */
  120. struct obs_hotkey {
  121. obs_hotkey_id id;
  122. char *name;
  123. char *description;
  124. obs_hotkey_func func;
  125. void *data;
  126. int pressed;
  127. obs_hotkey_registerer_t registerer_type;
  128. void *registerer;
  129. obs_hotkey_id pair_partner_id;
  130. UT_hash_handle hh;
  131. };
  132. struct obs_hotkey_pair {
  133. obs_hotkey_pair_id pair_id;
  134. obs_hotkey_id id[2];
  135. obs_hotkey_active_func func[2];
  136. bool pressed0;
  137. bool pressed1;
  138. void *data[2];
  139. UT_hash_handle hh;
  140. };
  141. typedef struct obs_hotkey_pair obs_hotkey_pair_t;
  142. typedef struct obs_hotkeys_platform obs_hotkeys_platform_t;
  143. void *obs_hotkey_thread(void *param);
  144. struct obs_core_hotkeys;
  145. bool obs_hotkeys_platform_init(struct obs_core_hotkeys *hotkeys);
  146. void obs_hotkeys_platform_free(struct obs_core_hotkeys *hotkeys);
  147. bool obs_hotkeys_platform_is_pressed(obs_hotkeys_platform_t *context,
  148. obs_key_t key);
  149. const char *obs_get_hotkey_translation(obs_key_t key, const char *def);
  150. struct obs_context_data;
  151. void obs_hotkeys_context_release(struct obs_context_data *context);
  152. void obs_hotkeys_free(void);
  153. struct obs_hotkey_binding {
  154. obs_key_combination_t key;
  155. bool pressed;
  156. bool modifiers_match;
  157. obs_hotkey_id hotkey_id;
  158. obs_hotkey_t *hotkey;
  159. };
  160. struct obs_hotkey_name_map_item;
  161. void obs_hotkey_name_map_free(void);
  162. /* ------------------------------------------------------------------------- */
  163. /* views */
  164. struct obs_view {
  165. pthread_mutex_t channels_mutex;
  166. obs_source_t *channels[MAX_CHANNELS];
  167. };
  168. extern bool obs_view_init(struct obs_view *view);
  169. extern void obs_view_free(struct obs_view *view);
  170. /* ------------------------------------------------------------------------- */
  171. /* displays */
  172. struct obs_display {
  173. bool update_color_space;
  174. bool enabled;
  175. uint32_t cx, cy;
  176. uint32_t next_cx, next_cy;
  177. uint32_t background_color;
  178. gs_swapchain_t *swap;
  179. pthread_mutex_t draw_callbacks_mutex;
  180. pthread_mutex_t draw_info_mutex;
  181. DARRAY(struct draw_callback) draw_callbacks;
  182. bool use_clear_workaround;
  183. struct obs_display *next;
  184. struct obs_display **prev_next;
  185. };
  186. extern bool obs_display_init(struct obs_display *display,
  187. const struct gs_init_data *graphics_data);
  188. extern void obs_display_free(struct obs_display *display);
  189. /* ------------------------------------------------------------------------- */
  190. /* core */
  191. struct obs_vframe_info {
  192. uint64_t timestamp;
  193. int count;
  194. };
  195. struct obs_tex_frame {
  196. gs_texture_t *tex;
  197. gs_texture_t *tex_uv;
  198. uint32_t handle;
  199. uint64_t timestamp;
  200. uint64_t lock_key;
  201. int count;
  202. bool released;
  203. };
  204. struct obs_task_info {
  205. obs_task_t task;
  206. void *param;
  207. };
  208. struct obs_core_video_mix {
  209. struct obs_view *view;
  210. gs_stagesurf_t *active_copy_surfaces[NUM_TEXTURES][NUM_CHANNELS];
  211. gs_stagesurf_t *copy_surfaces[NUM_TEXTURES][NUM_CHANNELS];
  212. gs_texture_t *convert_textures[NUM_CHANNELS];
  213. #ifdef _WIN32
  214. gs_stagesurf_t *copy_surfaces_encode[NUM_TEXTURES];
  215. gs_texture_t *convert_textures_encode[NUM_CHANNELS];
  216. #endif
  217. gs_texture_t *render_texture;
  218. gs_texture_t *output_texture;
  219. enum gs_color_space render_space;
  220. bool texture_rendered;
  221. bool textures_copied[NUM_TEXTURES];
  222. bool texture_converted;
  223. bool using_nv12_tex;
  224. bool using_p010_tex;
  225. struct circlebuf vframe_info_buffer;
  226. struct circlebuf vframe_info_buffer_gpu;
  227. gs_stagesurf_t *mapped_surfaces[NUM_CHANNELS];
  228. int cur_texture;
  229. volatile long raw_active;
  230. volatile long gpu_encoder_active;
  231. bool gpu_was_active;
  232. bool raw_was_active;
  233. bool was_active;
  234. pthread_mutex_t gpu_encoder_mutex;
  235. struct circlebuf gpu_encoder_queue;
  236. struct circlebuf gpu_encoder_avail_queue;
  237. DARRAY(obs_encoder_t *) gpu_encoders;
  238. os_sem_t *gpu_encode_semaphore;
  239. os_event_t *gpu_encode_inactive;
  240. pthread_t gpu_encode_thread;
  241. bool gpu_encode_thread_initialized;
  242. volatile bool gpu_encode_stop;
  243. video_t *video;
  244. struct obs_video_info ovi;
  245. bool gpu_conversion;
  246. const char *conversion_techs[NUM_CHANNELS];
  247. bool conversion_needed;
  248. float conversion_width_i;
  249. float conversion_height_i;
  250. float color_matrix[16];
  251. bool encoder_only_mix;
  252. long encoder_refs;
  253. };
  254. extern struct obs_core_video_mix *
  255. obs_create_video_mix(struct obs_video_info *ovi);
  256. extern void obs_free_video_mix(struct obs_core_video_mix *video);
  257. struct obs_core_video {
  258. graphics_t *graphics;
  259. gs_effect_t *default_effect;
  260. gs_effect_t *default_rect_effect;
  261. gs_effect_t *opaque_effect;
  262. gs_effect_t *solid_effect;
  263. gs_effect_t *repeat_effect;
  264. gs_effect_t *conversion_effect;
  265. gs_effect_t *bicubic_effect;
  266. gs_effect_t *lanczos_effect;
  267. gs_effect_t *area_effect;
  268. gs_effect_t *bilinear_lowres_effect;
  269. gs_effect_t *premultiplied_alpha_effect;
  270. gs_samplerstate_t *point_sampler;
  271. uint64_t video_time;
  272. uint64_t video_frame_interval_ns;
  273. uint64_t video_half_frame_interval_ns;
  274. uint64_t video_avg_frame_time_ns;
  275. double video_fps;
  276. pthread_t video_thread;
  277. uint32_t total_frames;
  278. uint32_t lagged_frames;
  279. bool thread_initialized;
  280. gs_texture_t *transparent_texture;
  281. gs_effect_t *deinterlace_discard_effect;
  282. gs_effect_t *deinterlace_discard_2x_effect;
  283. gs_effect_t *deinterlace_linear_effect;
  284. gs_effect_t *deinterlace_linear_2x_effect;
  285. gs_effect_t *deinterlace_blend_effect;
  286. gs_effect_t *deinterlace_blend_2x_effect;
  287. gs_effect_t *deinterlace_yadif_effect;
  288. gs_effect_t *deinterlace_yadif_2x_effect;
  289. float sdr_white_level;
  290. float hdr_nominal_peak_level;
  291. pthread_mutex_t task_mutex;
  292. struct circlebuf tasks;
  293. pthread_mutex_t mixes_mutex;
  294. DARRAY(struct obs_core_video_mix *) mixes;
  295. struct obs_core_video_mix *main_mix;
  296. };
  297. struct audio_monitor;
  298. struct obs_core_audio {
  299. audio_t *audio;
  300. DARRAY(struct obs_source *) render_order;
  301. DARRAY(struct obs_source *) root_nodes;
  302. uint64_t buffered_ts;
  303. struct circlebuf buffered_timestamps;
  304. uint64_t buffering_wait_ticks;
  305. int total_buffering_ticks;
  306. int max_buffering_ticks;
  307. bool fixed_buffer;
  308. pthread_mutex_t monitoring_mutex;
  309. DARRAY(struct audio_monitor *) monitors;
  310. char *monitoring_device_name;
  311. char *monitoring_device_id;
  312. pthread_mutex_t task_mutex;
  313. struct circlebuf tasks;
  314. };
  315. /* user sources, output channels, and displays */
  316. struct obs_core_data {
  317. /* Hash tables (uthash) */
  318. struct obs_source *sources; /* Lookup by UUID (hh_uuid) */
  319. struct obs_source *public_sources; /* Lookup by name (hh) */
  320. /* Linked lists */
  321. struct obs_source *first_audio_source;
  322. struct obs_display *first_display;
  323. struct obs_output *first_output;
  324. struct obs_encoder *first_encoder;
  325. struct obs_service *first_service;
  326. pthread_mutex_t sources_mutex;
  327. pthread_mutex_t displays_mutex;
  328. pthread_mutex_t outputs_mutex;
  329. pthread_mutex_t encoders_mutex;
  330. pthread_mutex_t services_mutex;
  331. pthread_mutex_t audio_sources_mutex;
  332. pthread_mutex_t draw_callbacks_mutex;
  333. DARRAY(struct draw_callback) draw_callbacks;
  334. DARRAY(struct rendered_callback) rendered_callbacks;
  335. DARRAY(struct tick_callback) tick_callbacks;
  336. struct obs_view main_view;
  337. long long unnamed_index;
  338. obs_data_t *private_data;
  339. volatile bool valid;
  340. DARRAY(char *) protocols;
  341. DARRAY(obs_source_t *) sources_to_tick;
  342. };
  343. /* user hotkeys */
  344. struct obs_core_hotkeys {
  345. pthread_mutex_t mutex;
  346. obs_hotkey_t *hotkeys;
  347. obs_hotkey_id next_id;
  348. obs_hotkey_pair_t *hotkey_pairs;
  349. obs_hotkey_pair_id next_pair_id;
  350. pthread_t hotkey_thread;
  351. bool hotkey_thread_initialized;
  352. os_event_t *stop_event;
  353. bool thread_disable_press;
  354. bool strict_modifiers;
  355. bool reroute_hotkeys;
  356. DARRAY(obs_hotkey_binding_t) bindings;
  357. obs_hotkey_callback_router_func router_func;
  358. void *router_func_data;
  359. obs_hotkeys_platform_t *platform_context;
  360. pthread_once_t name_map_init_token;
  361. struct obs_hotkey_name_map_item *name_map;
  362. signal_handler_t *signals;
  363. char *translations[OBS_KEY_LAST_VALUE];
  364. char *mute;
  365. char *unmute;
  366. char *push_to_mute;
  367. char *push_to_talk;
  368. char *sceneitem_show;
  369. char *sceneitem_hide;
  370. };
  371. struct obs_core {
  372. struct obs_module *first_module;
  373. DARRAY(struct obs_module_path) module_paths;
  374. DARRAY(struct obs_source_info) source_types;
  375. DARRAY(struct obs_source_info) input_types;
  376. DARRAY(struct obs_source_info) filter_types;
  377. DARRAY(struct obs_source_info) transition_types;
  378. DARRAY(struct obs_output_info) output_types;
  379. DARRAY(struct obs_encoder_info) encoder_types;
  380. DARRAY(struct obs_service_info) service_types;
  381. DARRAY(struct obs_modal_ui) modal_ui_callbacks;
  382. DARRAY(struct obs_modeless_ui) modeless_ui_callbacks;
  383. signal_handler_t *signals;
  384. proc_handler_t *procs;
  385. char *locale;
  386. char *module_config_path;
  387. bool name_store_owned;
  388. profiler_name_store_t *name_store;
  389. /* segmented into multiple sub-structures to keep things a bit more
  390. * clean and organized */
  391. struct obs_core_video video;
  392. struct obs_core_audio audio;
  393. struct obs_core_data data;
  394. struct obs_core_hotkeys hotkeys;
  395. os_task_queue_t *destruction_task_thread;
  396. obs_task_handler_t ui_task_handler;
  397. };
  398. extern struct obs_core *obs;
  399. struct obs_graphics_context {
  400. uint64_t last_time;
  401. uint64_t interval;
  402. uint64_t frame_time_total_ns;
  403. uint64_t fps_total_ns;
  404. uint32_t fps_total_frames;
  405. const char *video_thread_name;
  406. };
  407. extern void *obs_graphics_thread(void *param);
  408. extern bool obs_graphics_thread_loop(struct obs_graphics_context *context);
  409. #ifdef __APPLE__
  410. extern void *obs_graphics_thread_autorelease(void *param);
  411. extern bool
  412. obs_graphics_thread_loop_autorelease(struct obs_graphics_context *context);
  413. #endif
  414. extern gs_effect_t *obs_load_effect(gs_effect_t **effect, const char *file);
  415. extern bool audio_callback(void *param, uint64_t start_ts_in,
  416. uint64_t end_ts_in, uint64_t *out_ts,
  417. uint32_t mixers, struct audio_output_data *mixes);
  418. extern struct obs_core_video_mix *get_mix_for_video(video_t *video);
  419. extern void
  420. start_raw_video(video_t *video, const struct video_scale_info *conversion,
  421. void (*callback)(void *param, struct video_data *frame),
  422. void *param);
  423. extern void stop_raw_video(video_t *video,
  424. void (*callback)(void *param,
  425. struct video_data *frame),
  426. void *param);
  427. /* ------------------------------------------------------------------------- */
  428. /* obs shared context data */
  429. struct obs_weak_ref {
  430. volatile long refs;
  431. volatile long weak_refs;
  432. };
  433. struct obs_weak_object {
  434. struct obs_weak_ref ref;
  435. struct obs_context_data *object;
  436. };
  437. typedef void (*obs_destroy_cb)(void *obj);
  438. struct obs_context_data {
  439. char *name;
  440. const char *uuid;
  441. void *data;
  442. obs_data_t *settings;
  443. signal_handler_t *signals;
  444. proc_handler_t *procs;
  445. enum obs_obj_type type;
  446. struct obs_weak_object *control;
  447. obs_destroy_cb destroy;
  448. DARRAY(obs_hotkey_id) hotkeys;
  449. DARRAY(obs_hotkey_pair_id) hotkey_pairs;
  450. obs_data_t *hotkey_data;
  451. DARRAY(char *) rename_cache;
  452. pthread_mutex_t rename_cache_mutex;
  453. pthread_mutex_t *mutex;
  454. struct obs_context_data *next;
  455. struct obs_context_data **prev_next;
  456. UT_hash_handle hh;
  457. UT_hash_handle hh_uuid;
  458. bool private;
  459. };
  460. extern bool obs_context_data_init(struct obs_context_data *context,
  461. enum obs_obj_type type, obs_data_t *settings,
  462. const char *name, const char *uuid,
  463. obs_data_t *hotkey_data, bool private);
  464. extern void obs_context_init_control(struct obs_context_data *context,
  465. void *object, obs_destroy_cb destroy);
  466. extern void obs_context_data_free(struct obs_context_data *context);
  467. extern void obs_context_data_insert(struct obs_context_data *context,
  468. pthread_mutex_t *mutex, void *first);
  469. extern void obs_context_data_insert_name(struct obs_context_data *context,
  470. pthread_mutex_t *mutex, void *first);
  471. extern void obs_context_data_insert_uuid(struct obs_context_data *context,
  472. pthread_mutex_t *mutex,
  473. void *first_uuid);
  474. extern void obs_context_data_remove(struct obs_context_data *context);
  475. extern void obs_context_data_remove_name(struct obs_context_data *context,
  476. void *phead);
  477. extern void obs_context_data_remove_uuid(struct obs_context_data *context,
  478. void *puuid_head);
  479. extern void obs_context_wait(struct obs_context_data *context);
  480. extern void obs_context_data_setname(struct obs_context_data *context,
  481. const char *name);
  482. extern void obs_context_data_setname_ht(struct obs_context_data *context,
  483. const char *name, void *phead);
  484. /* ------------------------------------------------------------------------- */
  485. /* ref-counting */
  486. static inline void obs_ref_addref(struct obs_weak_ref *ref)
  487. {
  488. os_atomic_inc_long(&ref->refs);
  489. }
  490. static inline bool obs_ref_release(struct obs_weak_ref *ref)
  491. {
  492. return os_atomic_dec_long(&ref->refs) == -1;
  493. }
  494. static inline void obs_weak_ref_addref(struct obs_weak_ref *ref)
  495. {
  496. os_atomic_inc_long(&ref->weak_refs);
  497. }
  498. static inline bool obs_weak_ref_release(struct obs_weak_ref *ref)
  499. {
  500. return os_atomic_dec_long(&ref->weak_refs) == -1;
  501. }
  502. static inline bool obs_weak_ref_get_ref(struct obs_weak_ref *ref)
  503. {
  504. long owners = os_atomic_load_long(&ref->refs);
  505. while (owners > -1) {
  506. if (os_atomic_compare_exchange_long(&ref->refs, &owners,
  507. owners + 1)) {
  508. return true;
  509. }
  510. }
  511. return false;
  512. }
  513. static inline bool obs_weak_ref_expired(struct obs_weak_ref *ref)
  514. {
  515. long owners = os_atomic_load_long(&ref->refs);
  516. return owners < 0;
  517. }
  518. /* ------------------------------------------------------------------------- */
  519. /* sources */
  520. struct async_frame {
  521. struct obs_source_frame *frame;
  522. long unused_count;
  523. bool used;
  524. };
  525. enum audio_action_type {
  526. AUDIO_ACTION_VOL,
  527. AUDIO_ACTION_MUTE,
  528. AUDIO_ACTION_PTT,
  529. AUDIO_ACTION_PTM,
  530. };
  531. struct audio_action {
  532. uint64_t timestamp;
  533. enum audio_action_type type;
  534. union {
  535. float vol;
  536. bool set;
  537. };
  538. };
  539. struct obs_weak_source {
  540. struct obs_weak_ref ref;
  541. struct obs_source *source;
  542. };
  543. struct audio_cb_info {
  544. obs_source_audio_capture_t callback;
  545. void *param;
  546. };
  547. struct caption_cb_info {
  548. obs_source_caption_t callback;
  549. void *param;
  550. };
  551. struct obs_source {
  552. struct obs_context_data context;
  553. struct obs_source_info info;
  554. /* general exposed flags that can be set for the source */
  555. uint32_t flags;
  556. uint32_t default_flags;
  557. uint32_t last_obs_ver;
  558. /* indicates ownership of the info.id buffer */
  559. bool owns_info_id;
  560. /* signals to call the source update in the video thread */
  561. long defer_update_count;
  562. /* ensures show/hide are only called once */
  563. volatile long show_refs;
  564. /* ensures activate/deactivate are only called once */
  565. volatile long activate_refs;
  566. /* source is in the process of being destroyed */
  567. volatile long destroying;
  568. /* used to indicate that the source has been removed and all
  569. * references to it should be released (not exactly how I would prefer
  570. * to handle things but it's the best option) */
  571. bool removed;
  572. /* used to indicate if the source should show up when queried for user ui */
  573. bool temp_removed;
  574. bool active;
  575. bool showing;
  576. /* used to temporarily disable sources if needed */
  577. bool enabled;
  578. /* hint to allow sources to render more quickly */
  579. bool texcoords_centered;
  580. /* timing (if video is present, is based upon video) */
  581. volatile bool timing_set;
  582. volatile uint64_t timing_adjust;
  583. uint64_t resample_offset;
  584. uint64_t last_audio_ts;
  585. uint64_t next_audio_ts_min;
  586. uint64_t next_audio_sys_ts_min;
  587. uint64_t last_frame_ts;
  588. uint64_t last_sys_timestamp;
  589. bool async_rendered;
  590. /* audio */
  591. bool audio_failed;
  592. bool audio_pending;
  593. bool pending_stop;
  594. bool audio_active;
  595. bool user_muted;
  596. bool muted;
  597. struct obs_source *next_audio_source;
  598. struct obs_source **prev_next_audio_source;
  599. uint64_t audio_ts;
  600. struct circlebuf audio_input_buf[MAX_AUDIO_CHANNELS];
  601. size_t last_audio_input_buf_size;
  602. DARRAY(struct audio_action) audio_actions;
  603. float *audio_output_buf[MAX_AUDIO_MIXES][MAX_AUDIO_CHANNELS];
  604. float *audio_mix_buf[MAX_AUDIO_CHANNELS];
  605. struct resample_info sample_info;
  606. audio_resampler_t *resampler;
  607. pthread_mutex_t audio_actions_mutex;
  608. pthread_mutex_t audio_buf_mutex;
  609. pthread_mutex_t audio_mutex;
  610. pthread_mutex_t audio_cb_mutex;
  611. DARRAY(struct audio_cb_info) audio_cb_list;
  612. struct obs_audio_data audio_data;
  613. size_t audio_storage_size;
  614. uint32_t audio_mixers;
  615. float user_volume;
  616. float volume;
  617. int64_t sync_offset;
  618. int64_t last_sync_offset;
  619. float balance;
  620. /* async video data */
  621. gs_texture_t *async_textures[MAX_AV_PLANES];
  622. gs_texrender_t *async_texrender;
  623. struct obs_source_frame *cur_async_frame;
  624. bool async_gpu_conversion;
  625. enum video_format async_format;
  626. bool async_full_range;
  627. uint8_t async_trc;
  628. enum video_format async_cache_format;
  629. bool async_cache_full_range;
  630. uint8_t async_cache_trc;
  631. enum gs_color_format async_texture_formats[MAX_AV_PLANES];
  632. int async_channel_count;
  633. long async_rotation;
  634. bool async_flip;
  635. bool async_linear_alpha;
  636. bool async_active;
  637. bool async_update_texture;
  638. bool async_unbuffered;
  639. bool async_decoupled;
  640. struct obs_source_frame *async_preload_frame;
  641. DARRAY(struct async_frame) async_cache;
  642. DARRAY(struct obs_source_frame *) async_frames;
  643. pthread_mutex_t async_mutex;
  644. uint32_t async_width;
  645. uint32_t async_height;
  646. uint32_t async_cache_width;
  647. uint32_t async_cache_height;
  648. uint32_t async_convert_width[MAX_AV_PLANES];
  649. uint32_t async_convert_height[MAX_AV_PLANES];
  650. pthread_mutex_t caption_cb_mutex;
  651. DARRAY(struct caption_cb_info) caption_cb_list;
  652. /* async video deinterlacing */
  653. uint64_t deinterlace_offset;
  654. uint64_t deinterlace_frame_ts;
  655. gs_effect_t *deinterlace_effect;
  656. struct obs_source_frame *prev_async_frame;
  657. gs_texture_t *async_prev_textures[MAX_AV_PLANES];
  658. gs_texrender_t *async_prev_texrender;
  659. uint32_t deinterlace_half_duration;
  660. enum obs_deinterlace_mode deinterlace_mode;
  661. bool deinterlace_top_first;
  662. bool deinterlace_rendered;
  663. /* filters */
  664. struct obs_source *filter_parent;
  665. struct obs_source *filter_target;
  666. DARRAY(struct obs_source *) filters;
  667. pthread_mutex_t filter_mutex;
  668. gs_texrender_t *filter_texrender;
  669. enum obs_allow_direct_render allow_direct;
  670. bool rendering_filter;
  671. bool filter_bypass_active;
  672. /* sources specific hotkeys */
  673. obs_hotkey_pair_id mute_unmute_key;
  674. obs_hotkey_id push_to_mute_key;
  675. obs_hotkey_id push_to_talk_key;
  676. bool push_to_mute_enabled;
  677. bool push_to_mute_pressed;
  678. bool user_push_to_mute_pressed;
  679. bool push_to_talk_enabled;
  680. bool push_to_talk_pressed;
  681. bool user_push_to_talk_pressed;
  682. uint64_t push_to_mute_delay;
  683. uint64_t push_to_mute_stop_time;
  684. uint64_t push_to_talk_delay;
  685. uint64_t push_to_talk_stop_time;
  686. /* transitions */
  687. uint64_t transition_start_time;
  688. uint64_t transition_duration;
  689. pthread_mutex_t transition_tex_mutex;
  690. gs_texrender_t *transition_texrender[2];
  691. pthread_mutex_t transition_mutex;
  692. obs_source_t *transition_sources[2];
  693. float transition_manual_clamp;
  694. float transition_manual_torque;
  695. float transition_manual_target;
  696. float transition_manual_val;
  697. bool transitioning_video;
  698. bool transitioning_audio;
  699. bool transition_source_active[2];
  700. uint32_t transition_alignment;
  701. uint32_t transition_actual_cx;
  702. uint32_t transition_actual_cy;
  703. uint32_t transition_cx;
  704. uint32_t transition_cy;
  705. uint32_t transition_fixed_duration;
  706. bool transition_use_fixed_duration;
  707. enum obs_transition_mode transition_mode;
  708. enum obs_transition_scale_type transition_scale_type;
  709. struct matrix4 transition_matrices[2];
  710. /* color space */
  711. gs_texrender_t *color_space_texrender;
  712. struct audio_monitor *monitor;
  713. enum obs_monitoring_type monitoring_type;
  714. obs_data_t *private_settings;
  715. };
  716. extern struct obs_source_info *get_source_info(const char *id);
  717. extern struct obs_source_info *get_source_info2(const char *unversioned_id,
  718. uint32_t ver);
  719. extern bool obs_source_init_context(struct obs_source *source,
  720. obs_data_t *settings, const char *name,
  721. const char *uuid, obs_data_t *hotkey_data,
  722. bool private);
  723. extern bool obs_transition_init(obs_source_t *transition);
  724. extern void obs_transition_free(obs_source_t *transition);
  725. extern void obs_transition_tick(obs_source_t *transition, float t);
  726. extern void obs_transition_enum_sources(obs_source_t *transition,
  727. obs_source_enum_proc_t enum_callback,
  728. void *param);
  729. extern void obs_transition_save(obs_source_t *source, obs_data_t *data);
  730. extern void obs_transition_load(obs_source_t *source, obs_data_t *data);
  731. struct audio_monitor *audio_monitor_create(obs_source_t *source);
  732. void audio_monitor_reset(struct audio_monitor *monitor);
  733. extern void audio_monitor_destroy(struct audio_monitor *monitor);
  734. extern obs_source_t *
  735. obs_source_create_set_last_ver(const char *id, const char *name,
  736. const char *uuid, obs_data_t *settings,
  737. obs_data_t *hotkey_data, uint32_t last_obs_ver,
  738. bool is_private);
  739. extern void obs_source_destroy(struct obs_source *source);
  740. enum view_type {
  741. MAIN_VIEW,
  742. AUX_VIEW,
  743. };
  744. static inline void obs_source_dosignal(struct obs_source *source,
  745. const char *signal_obs,
  746. const char *signal_source)
  747. {
  748. struct calldata data;
  749. uint8_t stack[128];
  750. calldata_init_fixed(&data, stack, sizeof(stack));
  751. calldata_set_ptr(&data, "source", source);
  752. if (signal_obs && !source->context.private)
  753. signal_handler_signal(obs->signals, signal_obs, &data);
  754. if (signal_source)
  755. signal_handler_signal(source->context.signals, signal_source,
  756. &data);
  757. }
  758. /* maximum timestamp variance in nanoseconds */
  759. #define MAX_TS_VAR 2000000000ULL
  760. static inline bool frame_out_of_bounds(const obs_source_t *source, uint64_t ts)
  761. {
  762. if (ts < source->last_frame_ts)
  763. return ((source->last_frame_ts - ts) > MAX_TS_VAR);
  764. else
  765. return ((ts - source->last_frame_ts) > MAX_TS_VAR);
  766. }
  767. static inline enum gs_color_format
  768. convert_video_format(enum video_format format, enum video_trc trc)
  769. {
  770. switch (trc) {
  771. case VIDEO_TRC_PQ:
  772. case VIDEO_TRC_HLG:
  773. return GS_RGBA16F;
  774. default:
  775. switch (format) {
  776. case VIDEO_FORMAT_RGBA:
  777. return GS_RGBA;
  778. case VIDEO_FORMAT_BGRA:
  779. case VIDEO_FORMAT_I40A:
  780. case VIDEO_FORMAT_I42A:
  781. case VIDEO_FORMAT_YUVA:
  782. case VIDEO_FORMAT_AYUV:
  783. return GS_BGRA;
  784. case VIDEO_FORMAT_I010:
  785. case VIDEO_FORMAT_P010:
  786. case VIDEO_FORMAT_I210:
  787. case VIDEO_FORMAT_I412:
  788. case VIDEO_FORMAT_YA2L:
  789. case VIDEO_FORMAT_P216:
  790. case VIDEO_FORMAT_P416:
  791. case VIDEO_FORMAT_V210:
  792. return GS_RGBA16F;
  793. default:
  794. return GS_BGRX;
  795. }
  796. }
  797. }
  798. static inline enum gs_color_space convert_video_space(enum video_format format,
  799. enum video_trc trc)
  800. {
  801. enum gs_color_space space = GS_CS_SRGB;
  802. if (convert_video_format(format, trc) == GS_RGBA16F) {
  803. switch (trc) {
  804. case VIDEO_TRC_DEFAULT:
  805. case VIDEO_TRC_SRGB:
  806. space = GS_CS_SRGB_16F;
  807. break;
  808. case VIDEO_TRC_PQ:
  809. case VIDEO_TRC_HLG:
  810. space = GS_CS_709_EXTENDED;
  811. }
  812. }
  813. return space;
  814. }
  815. extern void obs_source_set_texcoords_centered(obs_source_t *source,
  816. bool centered);
  817. extern void obs_source_activate(obs_source_t *source, enum view_type type);
  818. extern void obs_source_deactivate(obs_source_t *source, enum view_type type);
  819. extern void obs_source_video_tick(obs_source_t *source, float seconds);
  820. extern float obs_source_get_target_volume(obs_source_t *source,
  821. obs_source_t *target);
  822. extern void obs_source_audio_render(obs_source_t *source, uint32_t mixers,
  823. size_t channels, size_t sample_rate,
  824. size_t size);
  825. extern void add_alignment(struct vec2 *v, uint32_t align, int cx, int cy);
  826. extern struct obs_source_frame *filter_async_video(obs_source_t *source,
  827. struct obs_source_frame *in);
  828. extern bool update_async_texture(struct obs_source *source,
  829. const struct obs_source_frame *frame,
  830. gs_texture_t *tex, gs_texrender_t *texrender);
  831. extern bool update_async_textures(struct obs_source *source,
  832. const struct obs_source_frame *frame,
  833. gs_texture_t *tex[MAX_AV_PLANES],
  834. gs_texrender_t *texrender);
  835. extern bool set_async_texture_size(struct obs_source *source,
  836. const struct obs_source_frame *frame);
  837. extern void remove_async_frame(obs_source_t *source,
  838. struct obs_source_frame *frame);
  839. extern void set_deinterlace_texture_size(obs_source_t *source);
  840. extern void deinterlace_process_last_frame(obs_source_t *source,
  841. uint64_t sys_time);
  842. extern void deinterlace_update_async_video(obs_source_t *source);
  843. extern void deinterlace_render(obs_source_t *s);
  844. /* ------------------------------------------------------------------------- */
  845. /* outputs */
  846. enum delay_msg {
  847. DELAY_MSG_PACKET,
  848. DELAY_MSG_START,
  849. DELAY_MSG_STOP,
  850. };
  851. struct delay_data {
  852. enum delay_msg msg;
  853. uint64_t ts;
  854. struct encoder_packet packet;
  855. };
  856. typedef void (*encoded_callback_t)(void *data, struct encoder_packet *packet);
  857. struct obs_weak_output {
  858. struct obs_weak_ref ref;
  859. struct obs_output *output;
  860. };
  861. #define CAPTION_LINE_CHARS (32)
  862. #define CAPTION_LINE_BYTES (4 * CAPTION_LINE_CHARS)
  863. struct caption_text {
  864. char text[CAPTION_LINE_BYTES + 1];
  865. double display_duration;
  866. struct caption_text *next;
  867. };
  868. struct pause_data {
  869. pthread_mutex_t mutex;
  870. uint64_t last_video_ts;
  871. uint64_t ts_start;
  872. uint64_t ts_end;
  873. uint64_t ts_offset;
  874. };
  875. extern bool video_pause_check(struct pause_data *pause, uint64_t timestamp);
  876. extern bool audio_pause_check(struct pause_data *pause, struct audio_data *data,
  877. size_t sample_rate);
  878. extern void pause_reset(struct pause_data *pause);
  879. struct obs_output {
  880. struct obs_context_data context;
  881. struct obs_output_info info;
  882. /* indicates ownership of the info.id buffer */
  883. bool owns_info_id;
  884. bool received_video[MAX_OUTPUT_VIDEO_ENCODERS];
  885. bool received_audio;
  886. volatile bool data_active;
  887. volatile bool end_data_capture_thread_active;
  888. int64_t video_offsets[MAX_OUTPUT_VIDEO_ENCODERS];
  889. int64_t audio_offsets[MAX_OUTPUT_AUDIO_ENCODERS];
  890. int64_t highest_audio_ts;
  891. int64_t highest_video_ts[MAX_OUTPUT_VIDEO_ENCODERS];
  892. pthread_t end_data_capture_thread;
  893. os_event_t *stopping_event;
  894. pthread_mutex_t interleaved_mutex;
  895. DARRAY(struct encoder_packet) interleaved_packets;
  896. int stop_code;
  897. int reconnect_retry_sec;
  898. int reconnect_retry_max;
  899. int reconnect_retries;
  900. uint32_t reconnect_retry_cur_msec;
  901. float reconnect_retry_exp;
  902. pthread_t reconnect_thread;
  903. os_event_t *reconnect_stop_event;
  904. volatile bool reconnecting;
  905. volatile bool reconnect_thread_active;
  906. uint32_t starting_drawn_count;
  907. uint32_t starting_lagged_count;
  908. uint32_t starting_frame_count;
  909. int total_frames;
  910. volatile bool active;
  911. volatile bool paused;
  912. video_t *video;
  913. audio_t *audio;
  914. obs_encoder_t *video_encoders[MAX_OUTPUT_VIDEO_ENCODERS];
  915. obs_encoder_t *audio_encoders[MAX_OUTPUT_AUDIO_ENCODERS];
  916. obs_service_t *service;
  917. size_t mixer_mask;
  918. struct pause_data pause;
  919. struct circlebuf audio_buffer[MAX_AUDIO_MIXES][MAX_AV_PLANES];
  920. uint64_t audio_start_ts;
  921. uint64_t video_start_ts;
  922. size_t audio_size;
  923. size_t planes;
  924. size_t sample_rate;
  925. size_t total_audio_frames;
  926. uint32_t scaled_width;
  927. uint32_t scaled_height;
  928. bool video_conversion_set;
  929. bool audio_conversion_set;
  930. struct video_scale_info video_conversion;
  931. struct audio_convert_info audio_conversion;
  932. pthread_mutex_t caption_mutex;
  933. double caption_timestamp;
  934. struct caption_text *caption_head;
  935. struct caption_text *caption_tail;
  936. struct circlebuf caption_data;
  937. bool valid;
  938. uint64_t active_delay_ns;
  939. encoded_callback_t delay_callback;
  940. struct circlebuf delay_data; /* struct delay_data */
  941. pthread_mutex_t delay_mutex;
  942. uint32_t delay_sec;
  943. uint32_t delay_flags;
  944. uint32_t delay_cur_flags;
  945. volatile long delay_restart_refs;
  946. volatile bool delay_active;
  947. volatile bool delay_capturing;
  948. char *last_error_message;
  949. float audio_data[MAX_AUDIO_CHANNELS][AUDIO_OUTPUT_FRAMES];
  950. };
  951. static inline void do_output_signal(struct obs_output *output,
  952. const char *signal)
  953. {
  954. struct calldata params = {0};
  955. calldata_set_ptr(&params, "output", output);
  956. signal_handler_signal(output->context.signals, signal, &params);
  957. calldata_free(&params);
  958. }
  959. extern void process_delay(void *data, struct encoder_packet *packet);
  960. extern void obs_output_cleanup_delay(obs_output_t *output);
  961. extern bool obs_output_delay_start(obs_output_t *output);
  962. extern void obs_output_delay_stop(obs_output_t *output);
  963. extern bool obs_output_actual_start(obs_output_t *output);
  964. extern void obs_output_actual_stop(obs_output_t *output, bool force,
  965. uint64_t ts);
  966. extern const struct obs_output_info *find_output(const char *id);
  967. extern void obs_output_remove_encoder(struct obs_output *output,
  968. struct obs_encoder *encoder);
  969. extern void
  970. obs_encoder_packet_create_instance(struct encoder_packet *dst,
  971. const struct encoder_packet *src);
  972. void obs_output_destroy(obs_output_t *output);
  973. /* ------------------------------------------------------------------------- */
  974. /* encoders */
  975. struct obs_weak_encoder {
  976. struct obs_weak_ref ref;
  977. struct obs_encoder *encoder;
  978. };
  979. struct encoder_callback {
  980. bool sent_first_packet;
  981. void (*new_packet)(void *param, struct encoder_packet *packet);
  982. void *param;
  983. };
  984. struct obs_encoder {
  985. struct obs_context_data context;
  986. struct obs_encoder_info info;
  987. /* allows re-routing to another encoder */
  988. struct obs_encoder_info orig_info;
  989. pthread_mutex_t init_mutex;
  990. uint32_t samplerate;
  991. size_t planes;
  992. size_t blocksize;
  993. size_t framesize;
  994. size_t framesize_bytes;
  995. size_t mixer_idx;
  996. /* OBS_SCALE_DISABLE indicates GPU scaling is disabled */
  997. enum obs_scale_type gpu_scale_type;
  998. uint32_t scaled_width;
  999. uint32_t scaled_height;
  1000. enum video_format preferred_format;
  1001. volatile bool active;
  1002. volatile bool paused;
  1003. bool initialized;
  1004. /* indicates ownership of the info.id buffer */
  1005. bool owns_info_id;
  1006. uint32_t timebase_num;
  1007. uint32_t timebase_den;
  1008. int64_t cur_pts;
  1009. struct circlebuf audio_input_buffer[MAX_AV_PLANES];
  1010. uint8_t *audio_output_buffer[MAX_AV_PLANES];
  1011. /* if a video encoder is paired with an audio encoder, make it start
  1012. * up at the specific timestamp. if this is the audio encoder,
  1013. * wait_for_video makes it wait until it's ready to sync up with
  1014. * video */
  1015. bool wait_for_video;
  1016. bool first_received;
  1017. struct obs_encoder *paired_encoder;
  1018. int64_t offset_usec;
  1019. uint64_t first_raw_ts;
  1020. uint64_t start_ts;
  1021. pthread_mutex_t outputs_mutex;
  1022. DARRAY(obs_output_t *) outputs;
  1023. bool destroy_on_stop;
  1024. /* stores the video/audio media output pointer. video_t *or audio_t **/
  1025. void *media;
  1026. pthread_mutex_t callbacks_mutex;
  1027. DARRAY(struct encoder_callback) callbacks;
  1028. struct pause_data pause;
  1029. const char *profile_encoder_encode_name;
  1030. char *last_error_message;
  1031. /* reconfigure encoder at next possible opportunity */
  1032. bool reconfigure_requested;
  1033. };
  1034. extern struct obs_encoder_info *find_encoder(const char *id);
  1035. extern bool obs_encoder_initialize(obs_encoder_t *encoder);
  1036. extern void obs_encoder_shutdown(obs_encoder_t *encoder);
  1037. extern void obs_encoder_start(obs_encoder_t *encoder,
  1038. void (*new_packet)(void *param,
  1039. struct encoder_packet *packet),
  1040. void *param);
  1041. extern void obs_encoder_stop(obs_encoder_t *encoder,
  1042. void (*new_packet)(void *param,
  1043. struct encoder_packet *packet),
  1044. void *param);
  1045. extern void obs_encoder_add_output(struct obs_encoder *encoder,
  1046. struct obs_output *output);
  1047. extern void obs_encoder_remove_output(struct obs_encoder *encoder,
  1048. struct obs_output *output);
  1049. extern bool start_gpu_encode(obs_encoder_t *encoder);
  1050. extern void stop_gpu_encode(obs_encoder_t *encoder);
  1051. extern bool do_encode(struct obs_encoder *encoder, struct encoder_frame *frame);
  1052. extern void send_off_encoder_packet(obs_encoder_t *encoder, bool success,
  1053. bool received, struct encoder_packet *pkt);
  1054. void obs_encoder_destroy(obs_encoder_t *encoder);
  1055. /* ------------------------------------------------------------------------- */
  1056. /* services */
  1057. struct obs_weak_service {
  1058. struct obs_weak_ref ref;
  1059. struct obs_service *service;
  1060. };
  1061. struct obs_service {
  1062. struct obs_context_data context;
  1063. struct obs_service_info info;
  1064. /* indicates ownership of the info.id buffer */
  1065. bool owns_info_id;
  1066. bool active;
  1067. bool destroy;
  1068. struct obs_output *output;
  1069. };
  1070. extern const struct obs_service_info *find_service(const char *id);
  1071. extern void obs_service_activate(struct obs_service *service);
  1072. extern void obs_service_deactivate(struct obs_service *service, bool remove);
  1073. extern bool obs_service_initialize(struct obs_service *service,
  1074. struct obs_output *output);
  1075. void obs_service_destroy(obs_service_t *service);
  1076. void obs_output_remove_encoder_internal(struct obs_output *output,
  1077. struct obs_encoder *encoder);