Gstreamer+python: добавление и удаление аудиоисточников во время работы конвейера
Я работаю над примером скрипта Python, первоначально найденного здесь: Добавление и удаление аудиоисточников в / из конвейера GStreamer на ходу. Цель состоит в том, чтобы сделать сценарий, такой как приведенный выше, способным вставлять и удалять источники звука во время работы конвейера, но с элементом аудиоконвертации между источником и сумматором. Это связано с тем, что в более общем случае Adder хочет, чтобы входящие потоки имели одинаковый формат.
Итак, вот код; мы создаем 2 генератора (зуммера). Первый издает тон 1000 Гц и ждет клавиши возврата. Второй - это сигнал с частотой 500 Гц, который суммируется с первым после нажатия клавиши. Опять же, при нажатии клавиши возврата слышен только второй генератор.
#!/usr/bin/python
import gobject;
gobject.threads_init()
import gst
# THE FOLLOWING FUNCTION IS A REWORK OF THE ORIGINAL, STILL DOING THE JOB
def create_raw_audiotest_signal(pipe, freq, adder):
# create buzzer of a given freq
buzzer = gst.element_factory_make("audiotestsrc","buzzer%d" % freq)
buzzer.set_property("freq",freq)
pipe.add(buzzer)
buzzersrc=buzzer.get_pad("src")
# Gather a request sink pad on the mixer
sinkpad=adder.get_request_pad("sink%d")
# .. and connect it to the buzzer
buzzersrc.link(sinkpad)
return buzzer, buzzersrc, sinkpad
# THIS IS A MODIFIED VERSION, NOT WORKING, THAT JUST PUTS AN AUDIOCONVERT
# ELEMENT BETWEEN THE GENERATOR AND THE ADDER.
def create_audiotest_signal_with_converter(pipe, freq, adder):
# create buzzer of a given freq
buzzer = gst.element_factory_make("audiotestsrc","buzzer%d" % freq)
buzzer.set_property("freq",freq)
# add a converter because adder wants inputs with the same format.
ac = gst.element_factory_make("audioconvert", "ac%d" % freq)
pipe.add(buzzer, ac)
# link the buzzer with the converter ...
buzzer.link(ac)
buzzersrc=buzzer.get_pad("src")
# Gather a request sink pad on the mixer
sinkpad=adder.get_request_pad("sink%d")
# and then the converter to the adder
ac.get_pad('src').link(sinkpad)
return buzzer, buzzersrc, sinkpad
if __name__ == "__main__":
# First create our pipeline
pipe = gst.Pipeline("mypipe")
# Create a software mixer with "Adder"
adder = gst.element_factory_make("adder","audiomixer")
pipe.add(adder)
# Create the first buzzer..
#buzzer1, buzzersrc1, sinkpad1 = create_raw_audiotest_signal(pipe, 1000, adder)
buzzer1, buzzersrc1, sinkpad1 = create_audiotest_signal_with_converter(pipe, 1000, adder)
# Add some output
output = gst.element_factory_make("autoaudiosink", "audio_out")
pipe.add(output)
adder.link(output)
# Start the playback
pipe.set_state(gst.STATE_PLAYING)
raw_input("1kHz test sound. Press <ENTER> to continue.")
# Get another generator
#buzzer2, buzzersrc2, sinkpad2 = create_raw_audiotest_signal(pipe, 500, adder)
buzzer2, buzzersrc2, sinkpad2 = create_audiotest_signal_with_converter(pipe, 500, adder)
# Start the second buzzer (other ways streaming stops because of starvation)
buzzer2.set_state(gst.STATE_PLAYING)
raw_input("1kHz + 500Hz test sound playing simoultenously. Press <ENTER> to continue.")
# Before removing a source, we must use pad blocking to prevent state changes
buzzersrc1.set_blocked(True)
# Stop the first buzzer
buzzer1.set_state(gst.STATE_NULL)
# Unlink from the mixer
buzzersrc1.unlink(sinkpad2)
# Release the mixers first sink pad
adder.release_request_pad(sinkpad1)
# Because here none of the Adder's sink pads block, streaming continues
raw_input("Only 500Hz test sound. Press <ENTER> to stop.")
Если вы используете create_raw_audiotest_signal вместо create_audiotest_signal_with_converter в обоих вызовах, конечно, это работает. Если вы используете смесь из двух, это работает, но с нежелательной дополнительной задержкой между ними. Наиболее интересным является случай, когда вы используете аудиоконверт в обоих вызовах, но gtk блокирует первый ключ возврата.
У кого-нибудь есть предложения? Что я делаю неправильно? Заранее спасибо.
1 ответ
Я сам нашел ответ, это было действительно просто... Я добавил другие компоненты, но они живут в процессе разработки и продолжают иметь статус независимой игры. Таким образом, решение состоит в том, что весь конвейер настроен на игру, которая, в свою очередь, устанавливает статус для всех детей.
pipe.set_state (gst.STATE_PLAYING)
вместо:
buzzer2.set_state (gst.STATE_PLAYING)
и это снова работает.