NPP_NewStream: для поиска для локального файла установлено значение 0 (false)

Я пытаюсь реализовать плагин NPAPI с возможностью потоковой передачи (NP_SEEK + NPN_RequestRead). Неважно, что я пытаюсь логическое NPBool seekable всегда установлен в 0 (ложь).

Я запускаю Firefox (iceweasel на Debian) из командной строки в локальном файле:

$ iceweasel test1.html

Если я присоединяю gdb к плагину npapi, вот что я вижу:

(gdb) 
#2  0x00007f7e9da54e14 in mozilla::plugins::BrowserStreamChild::StreamConstructed (this=0x7f7e925cf310, mimeType=..., seekable=<optimized out>, stype=<optimized out>)
    at /tmp/buildd/iceweasel-24.6.0esr/dom/plugins/ipc/BrowserStreamChild.cpp:62
62      &mStream, seekable, stype);
(gdb) 
#3  0x00007f7e9da5688e in mozilla::plugins::PluginInstanceChild::AnswerPBrowserStreamConstructor (this=<optimized out>, aActor=<optimized out>, url=..., 
    length=<optimized out>, lastmodified=<optimized out>, notifyData=<optimized out>, headers=..., mimeType=..., seekable=@0x7fff25ed51df: false, rv=0x7fff25ed51e0, 
    stype=0x7fff25ed51e2) at /tmp/buildd/iceweasel-24.6.0esr/dom/plugins/ipc/PluginInstanceChild.cpp:2285
2285              ->StreamConstructed(mimeType, seekable, stype);
(gdb) p seekable
$1 = (const bool &) @0x7fff25ed51df: false
(gdb) up
#4  0x00007f7e9da8f77f in mozilla::plugins::PPluginInstanceChild::OnCallReceived (this=0x7f7e925f2c00, __msg=..., __reply=@0x7fff25ed5470: 0x0)
    at /tmp/buildd/iceweasel-24.6.0esr/build-xulrunner/ipc/ipdl/PPluginInstanceChild.cpp:2479
warning: Source file is more recent than executable.
2479                if ((!(AnswerPBrowserStreamConstructor(actor, url, length, lastmodified, notifyData, headers, mimeType, seekable, (&(rv)), (&(stype)))))) {
(gdb) list -
2469                if ((!(actor))) {
2470                    return MsgValueError;
2471                }
2472                (actor)->mId = RegisterID(actor, (__handle).mId);
2473                (actor)->mManager = this;
2474                (actor)->mChannel = mChannel;
2475                (mManagedPBrowserStreamChild).InsertElementSorted(actor);
2476                (actor)->mState = mozilla::plugins::PBrowserStream::__Start;
2477    
2478                int32_t __id = mId;
(gdb) list -
2459                    FatalError("Error deserializing 'bool'");
2460                    return MsgValueError;
2461                }
2462                (__msg).EndRead(__iter);
2463                if ((!(PPluginInstance::Transition(mState, Trigger(Trigger::Send, PPluginInstance::Msg_PBrowserStreamConstructor__ID), (&(mState)))))) {
2464                    NS_WARNING("bad state transition!");
2465                }
2466                NPError rv;
2467                uint16_t stype;
2468                actor = AllocPBrowserStream(url, length, lastmodified, notifyData, headers, mimeType, seekable, (&(rv)), (&(stype)));
(gdb) list -
2449                }
2450                if ((!(Read((&(headers)), (&(__msg)), (&(__iter)))))) {
2451                    FatalError("Error deserializing 'nsCString'");
2452                    return MsgValueError;
2453                }
2454                if ((!(Read((&(mimeType)), (&(__msg)), (&(__iter)))))) {
2455                    FatalError("Error deserializing 'nsCString'");
2456                    return MsgValueError;
2457                }
2458                if ((!(Read((&(seekable)), (&(__msg)), (&(__iter)))))) {
(gdb) up
#5  0x00007f7e9da868f0 in mozilla::plugins::PPluginModuleChild::OnCallReceived (this=<optimized out>, __msg=..., __reply=@0x7fff25ed5470: 0x0)
    at /tmp/buildd/iceweasel-24.6.0esr/build-xulrunner/ipc/ipdl/PPluginModuleChild.cpp:1023
warning: Source file is more recent than executable.
1023            return (__routed)->OnCallReceived(__msg, __reply);
(gdb) list -
1013    PPluginModuleChild::OnCallReceived(
1014            const Message& __msg,
1015            Message*& __reply)
1016    {
1017        int32_t __route = (__msg).routing_id();
1018        if ((MSG_ROUTING_CONTROL) != (__route)) {
1019            ChannelListener* __routed = Lookup(__route);
1020            if ((!(__routed))) {
1021                return MsgRouteError;
1022            }
(gdb) bt

Если я копирую test1.html к /var/wwwи затем укажите на http://localhost/test1.html все работает как положено.

Однако в документации упоминается

seekable
Boolean indicating whether the stream is seekable:
true: Seekable. Stream supports random access through calls to NPN_RequestRead (for example, local files or HTTP servers that support byte-range requests).

1 ответ

Решение

Документация прямо врет.

  • seekable флаг в вызове NPP_NewStream
    • происходит из OnStartBinding
    • который вызывает в nsPluginStreamListenerPeer::IsSeekable
    • который просто возвращается nsPluginStreamListenerPeer::mSeekable,
  • Единственный раз mSeekable когда-либо установлен true когда ( источник)
    1. Поток http (https, spdy)
    2. Http ответ не имеет Content-Encoding
    3. Ответ http обеспечивает Content-Length,
    4. Http ответ имеет Accept-Ranges: bytes (опускание заголовка не поддерживается)
  • Для всех других типов потоков (вкл. file://) и потоки http, не соответствующие требованиям seekable флаг, следовательно, всегда false,

Более того, NPN_RequestRead реализован только для потоков HTTP, но на самом деле не заботится о seekable и более того, фактически не проверяет, возвращается ли сервер 206,

Заключение

Вы можете использовать только NP_SEEK потоки с http (https, spdy). Вот почему материал работает с http://localhost, но не из локального файла (file://).

Другие вопросы по тегам