?? pbc.c
字號:
{ mpeg_segment_t *_segment = _cdio_list_node_data (node); _segment->referenced = false; } /* start from non-rejected lists */ _CDIO_LIST_FOREACH (node, obj->pbc_list) { pbc_t *_pbc = _cdio_list_node_data (node); vcd_assert (_pbc->id != NULL); if (_pbc->rejected) continue; _vcd_pbc_mark_id (obj, _pbc->id); } /* collect flags */ _CDIO_LIST_FOREACH (node, obj->pbc_list) { pbc_t *_pbc = _cdio_list_node_data (node); if (!_pbc->referenced) vcd_warn ("PSD item '%s' is unreachable", _pbc->id); } _CDIO_LIST_FOREACH (node, obj->mpeg_sequence_list) { mpeg_sequence_t *_sequence = _cdio_list_node_data (node); if (!_sequence->referenced) vcd_warn ("sequence '%s' is not reachable by PBC", _sequence->id); } _CDIO_LIST_FOREACH (node, obj->mpeg_segment_list) { mpeg_segment_t *_segment = _cdio_list_node_data (node); if (!_segment->referenced) vcd_warn ("segment item '%s' is unreachable", _segment->id); }}void_vcd_pbc_node_write (const VcdObj *obj, const pbc_t *_pbc, void *buf, bool extended){ vcd_assert (obj != NULL); vcd_assert (_pbc != NULL); vcd_assert (buf != NULL); if (extended) vcd_assert (_vcd_obj_has_cap_p (obj, _CAP_PBC_X)); switch (_pbc->type) { case PBC_PLAYLIST: { _PsdPlayListDescriptor *_md = buf; CdioListNode *node; int n; _md->type = PSD_TYPE_PLAY_LIST; _md->noi = _cdio_list_length (_pbc->item_id_list); vcd_assert (_pbc->lid < 0x8000); _md->lid = uint16_to_be (_pbc->lid | (_pbc->rejected ? 0x8000 : 0)); _md->prev_ofs = uint16_to_be (_lookup_psd_offset (obj, _pbc->prev_id, extended)); _md->next_ofs = uint16_to_be (_lookup_psd_offset (obj, _pbc->next_id, extended)); _md->return_ofs = uint16_to_be (_lookup_psd_offset (obj, _pbc->retn_id, extended)); _md->ptime = uint16_to_be (rint (_pbc->playing_time * 15.0)); _md->wtime = _wtime (_pbc->wait_time); _md->atime = _wtime (_pbc->auto_pause_time); n = 0; _CDIO_LIST_FOREACH (node, _pbc->item_id_list) { const char *_id = _cdio_list_node_data (node); uint16_t _pin; if (_id) { _pin = _vcd_pbc_pin_lookup (obj, _id); if (!_pin) vcd_error ("PSD: referenced play item '%s' not found", _id); _md->itemid[n] = uint16_to_be (_pin); } else _md->itemid[n] = 0; /* play nothing */ n++; } } break; case PBC_SELECTION: { PsdSelectionListDescriptor_t *_md = buf; const unsigned int _nos = _cdio_list_length (_pbc->select_id_list); if (extended) _md->type = PSD_TYPE_EXT_SELECTION_LIST; else _md->type = PSD_TYPE_SELECTION_LIST; if (!IN (_pbc->bsn, 1, MAX_PBC_SELECTIONS)) vcd_error ("selection '%s': BSN (%d) not in range [1..%d]", _pbc->id, _pbc->bsn, MAX_PBC_SELECTIONS); if (!IN (_nos, 0, MAX_PBC_SELECTIONS)) vcd_error ("selection '%s': too many selections (%d > %d)", _pbc->id, _nos, MAX_PBC_SELECTIONS); if (_nos + _pbc->bsn > 100) vcd_error ("selection '%s': BSN + NOS (%d + %d) > 100", _pbc->id, _pbc->bsn, _nos); _md->bsn = _pbc->bsn; _md->nos = _nos; vcd_assert (sizeof (PsdSelectionListFlags) == 1); /* selection flags */ if (_vcd_obj_has_cap_p (obj, _CAP_4C_SVCD)) _md->flags.SelectionAreaFlag = true; else _md->flags.SelectionAreaFlag = false; _md->flags.CommandListFlag = false; vcd_assert (_pbc->lid < 0x8000); _md->lid = uint16_to_be (_pbc->lid | (_pbc->rejected ? 0x8000 : 0)); _md->prev_ofs = uint16_to_be (_lookup_psd_offset (obj, _pbc->prev_id, extended)); _md->next_ofs = uint16_to_be (_lookup_psd_offset (obj, _pbc->next_id, extended)); _md->return_ofs = uint16_to_be (_lookup_psd_offset (obj, _pbc->retn_id, extended)); switch (_pbc->selection_type) { case _SEL_NORMAL: _md->default_ofs = uint16_to_be (_lookup_psd_offset (obj, _pbc->default_id, extended)); break; case _SEL_MULTI_DEF: _md->default_ofs = uint16_to_be (PSD_OFS_MULTI_DEF); if (_pbc->default_id) vcd_warn ("ignoring default target '%s' for multi default selection '%s'", _pbc->default_id, _pbc->id); break; case _SEL_MULTI_DEF_NO_NUM: _md->default_ofs = uint16_to_be (PSD_OFS_MULTI_DEF_NO_NUM); if (_pbc->default_id) vcd_warn ("ignoring default target '%s' for multi default (w/o num) selection '%s'", _pbc->default_id, _pbc->id); break; default: vcd_assert_not_reached (); break; } _md->timeout_ofs = uint16_to_be (_lookup_psd_offset (obj, _pbc->timeout_id, extended)); _md->totime = _wtime (_pbc->timeout_time); if (_pbc->loop_count > 0x7f) vcd_warn ("loop count %d > 127", _pbc->loop_count); _md->loop = (_pbc->loop_count > 0x7f) ? 0x7f : _pbc->loop_count; if (_pbc->jump_delayed) _md->loop |= 0x80; /* timeout related sanity checks */ if (_pbc->loop_count > 0 && _pbc->timeout_time >= 0 && !_pbc->timeout_id && !_nos) vcd_warn ("PSD: selection '%s': neither timeout nor select target available, but neither loop count is infinite nor timeout wait time", _pbc->id); if (_pbc->timeout_id && (_pbc->timeout_time < 0 || _pbc->loop_count <= 0)) vcd_warn ("PSD: selection '%s': timeout target '%s' is never used due to loop count or timeout wait time given", _pbc->id, _pbc->timeout_id); if (_pbc->item_id) { const uint16_t _pin = _vcd_pbc_pin_lookup (obj, _pbc->item_id); if (!_pin) vcd_error ("PSD: referenced play item '%s' not found", _pbc->item_id); _md->itemid = uint16_to_be (_pin); } else _md->itemid = 0; /* play nothing */ /* sanity checks */ switch (_pbc->selection_type) { case _SEL_NORMAL: break; case _SEL_MULTI_DEF: case _SEL_MULTI_DEF_NO_NUM: if (_pbc->jump_delayed) vcd_warn ("selection '%s': jump timing shall be immediate", _pbc->id); if (_pbc->bsn != 1) vcd_error ("selection '%s': BSN != 1 for multi default selection", _pbc->id); /* checking NOS == NOE */ if (!_pbc->item_id) vcd_error ("selection '%s': play nothing play item not allowed for multidefault list", _pbc->id); { mpeg_sequence_t *_seq; if ((_seq = _vcd_obj_get_sequence_by_id ((VcdObj *) obj, _pbc->item_id)) || (_seq = _vcd_obj_get_sequence_by_entry_id ((VcdObj *) obj, _pbc->item_id))) { const unsigned _entries = _cdio_list_length (_seq->entry_list) + 1; if (_nos != _entries) vcd_error ("selection '%s': number of entrypoints" " (%d for sequence '%s') != number of selections (%d)", _pbc->id, _entries, _pbc->item_id, _nos); } else vcd_error ("selection '%s': play item '%s'" " is requried to be sequence or entry point" " item for multi default selecton", _pbc->id, _pbc->item_id); } break; default: vcd_assert_not_reached (); break; } /* fill selection array */ { CdioListNode *node = NULL; int idx = 0; idx = 0; _CDIO_LIST_FOREACH (node, _pbc->select_id_list) { const char *_id = _cdio_list_node_data (node); _md->ofs[idx] = uint16_to_be (_lookup_psd_offset (obj, _id, extended)); idx++; } } if (extended || _vcd_obj_has_cap_p (obj, _CAP_4C_SVCD)) { PsdSelectionListDescriptorExtended *_md2; CdioListNode *node; int n; /* append extended selection areas */ _md2 = (void *) &_md->ofs[_nos]; _set_area_helper (&_md2->next_area, _pbc->next_area, _pbc->id); _set_area_helper (&_md2->prev_area, _pbc->prev_area, _pbc->id); _set_area_helper (&_md2->return_area, _pbc->return_area, _pbc->id); _set_area_helper (&_md2->default_area, _pbc->default_area, _pbc->id); n = 0; if (_pbc->select_area_list) _CDIO_LIST_FOREACH (node, _pbc->select_area_list) { const pbc_area_t *_area = _cdio_list_node_data (node); _set_area_helper (&_md2->area[n], _area, _pbc->id); n++; } vcd_assert (n == _nos); } } break; case PBC_END: { PsdEndListDescriptor *_md = buf; _md->type = PSD_TYPE_END_LIST; if (_vcd_obj_has_cap_p (obj, _CAP_4C_SVCD)) { _md->next_disc = _pbc->next_disc; if (_pbc->image_id) { uint16_t _pin = _vcd_pbc_pin_lookup (obj, _pbc->image_id); mpeg_segment_t *_segment; if (!_pbc->next_disc) vcd_warn ("PSD: endlist '%s': change disc picture given," " but next volume is 0", _pbc->id); if (!_pin) vcd_error ("PSD: referenced play item '%s' not found", _pbc->item_id); _md->change_pic = uint16_to_be (_pin); /* sanity checks */ _segment = _vcd_obj_get_segment_by_id ((VcdObj *) obj, _pbc->image_id); if (!_segment) vcd_warn ("PSD: endlist '%s': referenced play item '%s'" " is not a segment play item", _pbc->id, _pbc->image_id); else if (_segment->info->shdr[0].seen || !(_segment->info->shdr[1].seen || _segment->info->shdr[2].seen)) vcd_warn ("PSD: endlist '%s': referenced play item '%s'" " should be a still picture", _pbc->id, _pbc->image_id); } } else if (_pbc->next_disc || _pbc->image_id) vcd_warn ("extended end list attributes ignored for non-SVCD"); } break; default: vcd_assert_not_reached (); break; }}pbc_t *vcd_pbc_new (enum pbc_type_t type){ pbc_t *_pbc; _pbc = _vcd_malloc (sizeof (pbc_t)); _pbc->type = type; switch (type) { case PBC_PLAYLIST: _pbc->item_id_list = _cdio_list_new (); break; case PBC_SELECTION: _pbc->select_id_list = _cdio_list_new (); _pbc->select_area_list = _cdio_list_new (); break; case PBC_END: break; default: vcd_assert_not_reached (); break; } return _pbc;}/* */bool_vcd_pbc_finalize (VcdObj *obj){ CdioListNode *node; unsigned offset = 0, offset_ext = 0; unsigned lid; lid = 1; _CDIO_LIST_FOREACH (node, obj->pbc_list) { pbc_t *_pbc = _cdio_list_node_data (node); unsigned length, length_ext = 0; length = _vcd_pbc_node_length (obj, _pbc, false); if (_vcd_obj_has_cap_p (obj, _CAP_PBC_X)) length_ext = _vcd_pbc_node_length (obj, _pbc, true); /* round them up to... */ length = _vcd_ceil2block (length, INFO_OFFSET_MULT); if (_vcd_obj_has_cap_p (obj, _CAP_PBC_X)) length_ext = _vcd_ceil2block (length_ext, INFO_OFFSET_MULT); /* node may not cross sector boundary! */ offset = _vcd_ofs_add (offset, length, ISO_BLOCKSIZE); if (_vcd_obj_has_cap_p (obj, _CAP_PBC_X)) offset_ext = _vcd_ofs_add (offset_ext, length_ext, ISO_BLOCKSIZE); /* save start offsets */ _pbc->offset = offset - length; if (_vcd_obj_has_cap_p (obj, _CAP_PBC_X)) _pbc->offset_ext = offset_ext - length_ext; _pbc->lid = lid; lid++; } obj->psd_size = offset; if (_vcd_obj_has_cap_p (obj, _CAP_PBC_X)) obj->psdx_size = offset_ext; vcd_debug ("pbc: psd size %d (extended psd %d)", offset, offset_ext); return true;}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -