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

#firefox #npapi

#firefox #npapi

Вопрос:

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

Я запускаю 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      amp;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 amp;) @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, (amp;(rv)), (amp;(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), (amp;(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, (amp;(rv)), (amp;(stype)));
(gdb) list -
2449                }
2450                if ((!(Read((amp;(headers)), (amp;(__msg)), (amp;(__iter)))))) {
2451                    FatalError("Error deserializing 'nsCString'");
2452                    return MsgValueError;
2453                }
2454                if ((!(Read((amp;(mimeType)), (amp;(__msg)), (amp;(__iter)))))) {
2455                    FatalError("Error deserializing 'nsCString'");
2456                    return MsgValueError;
2457                }
2458                if ((!(Read((amp;(seekable)), (amp;(__msg)), (amp;(__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 Messageamp; __msg,
1015            Message*amp; __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
  • Устанавливается только время 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:// ).