texture-amf-opts.hpp 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. static bool str_to_bool(const char *str)
  2. {
  3. if (!str)
  4. return false;
  5. if (*str == '1')
  6. return true;
  7. if (*str == '0')
  8. return false;
  9. if (astrcmpi(str, "true") == 0)
  10. return true;
  11. if (astrcmpi(str, "false") == 0)
  12. return false;
  13. return false;
  14. }
  15. static void amf_apply_opt(amf_base *enc, obs_option *opt)
  16. {
  17. bool avc = enc->codec == amf_codec_type::AVC;
  18. bool hevc = enc->codec == amf_codec_type::HEVC;
  19. if (strcmp(opt->name, "g") == 0 || strcmp(opt->name, "keyint") == 0) {
  20. int val = atoi(opt->value);
  21. if (enc->codec == amf_codec_type::AVC)
  22. set_avc_opt(IDR_PERIOD, val);
  23. else
  24. set_hevc_opt(NUM_GOPS_PER_IDR, val);
  25. } else if (strcmp(opt->name, "usage") == 0) {
  26. if (strcmp(opt->value, "transcoding") == 0) {
  27. set_enum_opt(USAGE, TRANSCONDING);
  28. } else if (strcmp(opt->value, "ultralowlatency") == 0) {
  29. set_enum_opt(USAGE, ULTRA_LOW_LATENCY);
  30. } else if (strcmp(opt->value, "lowlatency") == 0) {
  31. set_enum_opt(USAGE, LOW_LATENCY);
  32. } else if (strcmp(opt->value, "webcam") == 0) {
  33. set_enum_opt(USAGE, WEBCAM);
  34. } else {
  35. warn("Invalid value for %s: %s", opt->name, opt->value);
  36. }
  37. } else if (strcmp(opt->name, "profile") == 0) {
  38. if (strcmp(opt->value, "main") == 0) {
  39. set_enum_opt(PROFILE, MAIN);
  40. } else if (enc->codec != amf_codec_type::AVC) {
  41. warn("Invalid value for %s: %s", opt->name, opt->value);
  42. return;
  43. }
  44. if (strcmp(opt->value, "high") == 0) {
  45. set_opt(PROFILE, AMF_VIDEO_ENCODER_PROFILE_HIGH);
  46. } else if (strcmp(opt->value, "constrained_baseline") == 0) {
  47. set_opt(PROFILE,
  48. AMF_VIDEO_ENCODER_PROFILE_CONSTRAINED_BASELINE);
  49. } else if (strcmp(opt->value, "constrained_high") == 0) {
  50. set_opt(PROFILE,
  51. AMF_VIDEO_ENCODER_PROFILE_CONSTRAINED_HIGH);
  52. } else {
  53. warn("Invalid value for %s: %s", opt->name, opt->value);
  54. }
  55. } else if (strcmp(opt->name, "level") == 0) {
  56. std::string val = opt->value;
  57. size_t pos = val.find('.');
  58. if (pos != std::string::npos)
  59. val.erase(pos, 1);
  60. int level = std::stoi(val);
  61. set_opt(PROFILE_LEVEL, level);
  62. } else if (strcmp(opt->name, "quality") == 0) {
  63. if (strcmp(opt->value, "speed") == 0) {
  64. set_enum_opt(QUALITY_PRESET, SPEED);
  65. } else if (strcmp(opt->value, "balanced") == 0) {
  66. set_enum_opt(QUALITY_PRESET, BALANCED);
  67. } else if (strcmp(opt->value, "quality") == 0) {
  68. set_enum_opt(QUALITY_PRESET, QUALITY);
  69. } else {
  70. warn("Invalid value for %s: %s", opt->name, opt->value);
  71. }
  72. } else if (strcmp(opt->name, "rc") == 0) {
  73. if (strcmp(opt->value, "cqp") == 0) {
  74. set_enum_opt(RATE_CONTROL_METHOD, CONSTANT_QP);
  75. } else if (strcmp(opt->value, "cbr") == 0) {
  76. set_enum_opt(RATE_CONTROL_METHOD, CBR);
  77. } else if (strcmp(opt->value, "vbr_peak") == 0) {
  78. set_enum_opt(RATE_CONTROL_METHOD, PEAK_CONSTRAINED_VBR);
  79. } else if (strcmp(opt->value, "vbr_latency") == 0) {
  80. set_enum_opt(RATE_CONTROL_METHOD,
  81. LATENCY_CONSTRAINED_VBR);
  82. } else {
  83. warn("Invalid value for %s: %s", opt->name, opt->value);
  84. }
  85. } else if (strcmp(opt->name, "enforce_hrd") == 0) {
  86. bool val = str_to_bool(opt->value);
  87. set_opt(ENFORCE_HRD, val);
  88. } else if (strcmp(opt->name, "filler_data") == 0) {
  89. bool val = str_to_bool(opt->value);
  90. set_opt(FILLER_DATA_ENABLE, val);
  91. } else if (strcmp(opt->name, "vbaq") == 0) {
  92. bool val = str_to_bool(opt->value);
  93. set_opt(ENABLE_VBAQ, val);
  94. } else if (strcmp(opt->name, "qp_i") == 0) {
  95. int val = atoi(opt->value);
  96. set_opt(QP_I, val);
  97. } else if (strcmp(opt->name, "qp_p") == 0) {
  98. int val = atoi(opt->value);
  99. set_opt(QP_P, val);
  100. } else if (strcmp(opt->name, "me_half_pel") == 0) {
  101. bool val = str_to_bool(opt->value);
  102. set_opt(MOTION_HALF_PIXEL, val);
  103. } else if (strcmp(opt->name, "me_quarter_pel") == 0) {
  104. bool val = str_to_bool(opt->value);
  105. set_opt(MOTION_QUARTERPIXEL, val);
  106. } else if (strcmp(opt->name, "aud") == 0) {
  107. bool val = str_to_bool(opt->value);
  108. set_opt(INSERT_AUD, val);
  109. } else if (strcmp(opt->name, "max_au_size") == 0) {
  110. int val = atoi(opt->value);
  111. set_opt(MAX_AU_SIZE, val);
  112. } else if (avc && strcmp(opt->name, "preanalysis") == 0) {
  113. bool val = str_to_bool(opt->value);
  114. set_avc_property(enc, PREENCODE_ENABLE, val);
  115. } else if (avc && strcmp(opt->name, "qp_b") == 0) {
  116. int val = atoi(opt->value);
  117. set_avc_property(enc, QP_B, val);
  118. } else if (avc && strcmp(opt->name, "frame_skipping") == 0) {
  119. bool val = str_to_bool(opt->value);
  120. set_avc_property(enc, RATE_CONTROL_SKIP_FRAME_ENABLE, val);
  121. } else if (avc && strcmp(opt->name, "header_spacing") == 0) {
  122. int val = atoi(opt->value);
  123. set_avc_property(enc, HEADER_INSERTION_SPACING, val);
  124. } else if (avc && strcmp(opt->name, "bf_delta_qp") == 0) {
  125. int val = atoi(opt->value);
  126. set_avc_property(enc, B_PIC_DELTA_QP, val);
  127. } else if (avc && strcmp(opt->name, "bf_ref") == 0) {
  128. bool val = str_to_bool(opt->value);
  129. set_avc_property(enc, B_REFERENCE_ENABLE, val);
  130. } else if (avc && strcmp(opt->name, "bf_ref_delta_qp") == 0) {
  131. int val = atoi(opt->value);
  132. set_avc_property(enc, REF_B_PIC_DELTA_QP, val);
  133. } else if (avc && strcmp(opt->name, "intra_refresh_mb") == 0) {
  134. int val = atoi(opt->value);
  135. set_avc_property(enc, INTRA_REFRESH_NUM_MBS_PER_SLOT, val);
  136. } else if (avc && strcmp(opt->name, "coder") == 0) {
  137. if (strcmp(opt->value, "auto") == 0) {
  138. set_avc_opt(CABAC_ENABLE, AMF_VIDEO_ENCODER_UNDEFINED);
  139. } else if (strcmp(opt->value, "cavlc") == 0) {
  140. set_avc_opt(CABAC_ENABLE, AMF_VIDEO_ENCODER_CALV);
  141. } else if (strcmp(opt->value, "cabac") == 0) {
  142. set_avc_opt(CABAC_ENABLE, AMF_VIDEO_ENCODER_CABAC);
  143. } else {
  144. warn("Invalid value for %s: %s", opt->name, opt->value);
  145. }
  146. } else if (hevc && strcmp(opt->name, "profile_tier") == 0) {
  147. if (strcmp(opt->value, "main") == 0) {
  148. set_hevc_enum(TIER, MAIN);
  149. } else if (strcmp(opt->value, "high") == 0) {
  150. set_hevc_enum(TIER, HIGH);
  151. } else {
  152. warn("Invalid value for %s: %s", opt->name, opt->value);
  153. }
  154. } else if (hevc && strcmp(opt->name, "header_insertion_mode") == 0) {
  155. if (strcmp(opt->value, "none") == 0) {
  156. set_hevc_enum(HEADER_INSERTION_MODE, NONE);
  157. } else if (strcmp(opt->value, "gop") == 0) {
  158. set_hevc_enum(HEADER_INSERTION_MODE, GOP_ALIGNED);
  159. } else if (strcmp(opt->value, "idr") == 0) {
  160. set_hevc_enum(HEADER_INSERTION_MODE, IDR_ALIGNED);
  161. } else {
  162. warn("Invalid value for %s: %s", opt->name, opt->value);
  163. }
  164. } else if (hevc && strcmp(opt->name, "skip_frame") == 0) {
  165. bool val = str_to_bool(opt->value);
  166. set_hevc_property(enc, RATE_CONTROL_SKIP_FRAME_ENABLE, val);
  167. } else if (hevc && strcmp(opt->name, "gops_per_idr") == 0) {
  168. int val = atoi(opt->value);
  169. set_hevc_property(enc, NUM_GOPS_PER_IDR, val);
  170. } else if (hevc && strcmp(opt->name, "min_qp_i") == 0) {
  171. int val = atoi(opt->value);
  172. set_hevc_property(enc, MIN_QP_I, val);
  173. } else if (hevc && strcmp(opt->name, "max_qp_i") == 0) {
  174. int val = atoi(opt->value);
  175. set_hevc_property(enc, MAX_QP_I, val);
  176. } else if (hevc && strcmp(opt->name, "min_qp_i") == 0) {
  177. int val = atoi(opt->value);
  178. set_hevc_property(enc, MIN_QP_P, val);
  179. } else if (hevc && strcmp(opt->name, "max_qp_i") == 0) {
  180. int val = atoi(opt->value);
  181. set_hevc_property(enc, MAX_QP_P, val);
  182. } else {
  183. wchar_t wname[256];
  184. int val;
  185. bool is_bool = false;
  186. if (astrcmpi(opt->value, "true") == 0) {
  187. is_bool = true;
  188. val = 1;
  189. } else if (astrcmpi(opt->value, "false") == 0) {
  190. is_bool = true;
  191. val = 0;
  192. } else {
  193. val = atoi(opt->value);
  194. }
  195. os_utf8_to_wcs(opt->name, 0, wname, _countof(wname));
  196. if (is_bool) {
  197. bool bool_val = (bool)val;
  198. set_amf_property(enc, wname, bool_val);
  199. } else {
  200. set_amf_property(enc, wname, val);
  201. }
  202. }
  203. }