raw.c 13.2 KB
Newer Older
F
Fabrice Bellard 已提交
1 2
/* 
 * RAW encoder and decoder
F
Fabrice Bellard 已提交
3
 * Copyright (c) 2001 Fabrice Bellard.
F
Fabrice Bellard 已提交
4
 *
F
Fabrice Bellard 已提交
5 6 7 8
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) any later version.
F
Fabrice Bellard 已提交
9
 *
F
Fabrice Bellard 已提交
10
 * This library is distributed in the hope that it will be useful,
F
Fabrice Bellard 已提交
11
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
F
Fabrice Bellard 已提交
12 13
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
F
Fabrice Bellard 已提交
14
 *
F
Fabrice Bellard 已提交
15 16 17
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
F
Fabrice Bellard 已提交
18 19 20 21
 */
#include "avformat.h"

/* simple formats */
22
static int raw_write_header(struct AVFormatContext *s)
F
Fabrice Bellard 已提交
23 24 25 26
{
    return 0;
}

27 28
static int raw_write_packet(struct AVFormatContext *s, int stream_index,
			    unsigned char *buf, int size, int force_pts)
F
Fabrice Bellard 已提交
29 30 31 32 33 34
{
    put_buffer(&s->pb, buf, size);
    put_flush_packet(&s->pb);
    return 0;
}

35
static int raw_write_trailer(struct AVFormatContext *s)
F
Fabrice Bellard 已提交
36 37 38 39 40
{
    return 0;
}

/* raw input */
41
static int raw_read_header(AVFormatContext *s, AVFormatParameters *ap)
F
Fabrice Bellard 已提交
42 43
{
    AVStream *st;
F
Fabrice Bellard 已提交
44
    int id;
F
Fabrice Bellard 已提交
45

F
Fabrice Bellard 已提交
46
    st = av_new_stream(s, 0);
F
Fabrice Bellard 已提交
47
    if (!st)
F
Fabrice Bellard 已提交
48
        return AVERROR_NOMEM;
F
Fabrice Bellard 已提交
49
    if (ap) {
F
Fabrice Bellard 已提交
50 51
        id = s->iformat->value;
        if (id == CODEC_ID_RAWVIDEO) {
F
Fabrice Bellard 已提交
52 53
            st->codec.codec_type = CODEC_TYPE_VIDEO;
        } else {
F
Fabrice Bellard 已提交
54
            st->codec.codec_type = CODEC_TYPE_AUDIO;
F
Fabrice Bellard 已提交
55
        }
F
Fabrice Bellard 已提交
56 57
        st->codec.codec_id = id;

F
Fabrice Bellard 已提交
58 59 60 61 62 63
        switch(st->codec.codec_type) {
        case CODEC_TYPE_AUDIO:
            st->codec.sample_rate = ap->sample_rate;
            st->codec.channels = ap->channels;
            break;
        case CODEC_TYPE_VIDEO:
64 65
            st->codec.frame_rate      = ap->frame_rate;
            st->codec.frame_rate_base = ap->frame_rate_base;
F
Fabrice Bellard 已提交
66 67
            st->codec.width = ap->width;
            st->codec.height = ap->height;
68
	    st->codec.pix_fmt = ap->pix_fmt;
F
Fabrice Bellard 已提交
69 70
            break;
        default:
F
Fabrice Bellard 已提交
71
            return -1;
F
Fabrice Bellard 已提交
72 73
        }
    } else {
F
Fabrice Bellard 已提交
74
        return -1;
F
Fabrice Bellard 已提交
75 76 77 78
    }
    return 0;
}

F
Fabrice Bellard 已提交
79
#define RAW_PACKET_SIZE 1024
F
Fabrice Bellard 已提交
80

81
static int raw_read_packet(AVFormatContext *s, AVPacket *pkt)
F
Fabrice Bellard 已提交
82
{
M
Michael Niedermayer 已提交
83
    int ret, size;
F
Fabrice Bellard 已提交
84
    //    AVStream *st = s->streams[0];
M
Michael Niedermayer 已提交
85
    
86
    size= RAW_PACKET_SIZE;
F
Fabrice Bellard 已提交
87

M
Michael Niedermayer 已提交
88
    if (av_new_packet(pkt, size) < 0)
F
Fabrice Bellard 已提交
89 90 91
        return -EIO;

    pkt->stream_index = 0;
M
Michael Niedermayer 已提交
92
    ret = get_buffer(&s->pb, pkt->data, size);
F
Fabrice Bellard 已提交
93
    if (ret <= 0) {
F
Fabrice Bellard 已提交
94
        av_free_packet(pkt);
F
Fabrice Bellard 已提交
95 96 97 98 99
        return -EIO;
    }
    /* note: we need to modify the packet size here to handle the last
       packet */
    pkt->size = ret;
F
Fabrice Bellard 已提交
100 101 102
    return ret;
}

103
static int raw_read_close(AVFormatContext *s)
F
Fabrice Bellard 已提交
104 105 106 107 108 109 110 111 112
{
    return 0;
}

/* mp3 read */
static int mp3_read_header(AVFormatContext *s,
                           AVFormatParameters *ap)
{
    AVStream *st;
113
    int pos;
F
Fabrice Bellard 已提交
114

F
Fabrice Bellard 已提交
115
    st = av_new_stream(s, 0);
F
Fabrice Bellard 已提交
116
    if (!st)
F
Fabrice Bellard 已提交
117
        return AVERROR_NOMEM;
F
Fabrice Bellard 已提交
118 119 120

    st->codec.codec_type = CODEC_TYPE_AUDIO;
    st->codec.codec_id = CODEC_ID_MP2;
121 122 123 124 125 126 127 128 129 130 131 132 133

    /* looking for 11111111 111MMLLC - MPEG synchronization tag
	MM: 00 - MPEG-2.5, 10 - MPEG-2, 11 - MPEG-1
	LL: 11 - Layer I, 10 - Layer II, 01 - Layer III
	XXX: this code does not read more bytes from file 
	so if ID3 (or other stuff) length > IO_BUFFER_SIZE it fails back to CODEC_ID_MP2 */
    for(pos=0; pos < s->pb.buffer_size-1; pos++)
	if( s->pb.buffer[pos] == 0xFF && (s->pb.buffer[pos] & 0xE0) == 0xE0 )
	    break;
	    
    if( pos < s->pb.buffer_size-1 && (s->pb.buffer[pos+1] & 6) == 2 )
	st->codec.codec_id = CODEC_ID_MP3LAME;

F
Fabrice Bellard 已提交
134
    /* the parameters will be extracted from the compressed bitstream */
F
Fabrice Bellard 已提交
135 136 137
    return 0;
}

138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153
/* ac3 read */
static int ac3_read_header(AVFormatContext *s,
                           AVFormatParameters *ap)
{
    AVStream *st;

    st = av_new_stream(s, 0);
    if (!st)
        return AVERROR_NOMEM;

    st->codec.codec_type = CODEC_TYPE_AUDIO;
    st->codec.codec_id = CODEC_ID_AC3;
    /* the parameters will be extracted from the compressed bitstream */
    return 0;
}

F
Fabrice Bellard 已提交
154 155 156 157 158 159
/* mpeg1/h263 input */
static int video_read_header(AVFormatContext *s,
                             AVFormatParameters *ap)
{
    AVStream *st;

F
Fabrice Bellard 已提交
160
    st = av_new_stream(s, 0);
F
Fabrice Bellard 已提交
161
    if (!st)
F
Fabrice Bellard 已提交
162
        return AVERROR_NOMEM;
F
Fabrice Bellard 已提交
163 164

    st->codec.codec_type = CODEC_TYPE_VIDEO;
F
Fabrice Bellard 已提交
165
    st->codec.codec_id = s->iformat->value;
F
Fabrice Bellard 已提交
166
    /* for mjpeg, specify frame rate */
M
Michael Niedermayer 已提交
167 168
    /* for mpeg4 specify it too (most mpeg4 streams dont have the fixed_vop_rate set ...)*/
    if (st->codec.codec_id == CODEC_ID_MJPEG || st->codec.codec_id == CODEC_ID_MPEG4) {
F
Fabrice Bellard 已提交
169
        if (ap) {
170 171
            st->codec.frame_rate      = ap->frame_rate;
            st->codec.frame_rate_base = ap->frame_rate_base;
F
Fabrice Bellard 已提交
172
        } else {
173 174
            st->codec.frame_rate      = 25;
            st->codec.frame_rate_base = 1;
F
Fabrice Bellard 已提交
175 176
        }
    }
F
Fabrice Bellard 已提交
177 178 179
    return 0;
}

F
Fabrice Bellard 已提交
180 181 182 183 184 185 186
#define SEQ_START_CODE		0x000001b3
#define GOP_START_CODE		0x000001b8
#define PICTURE_START_CODE	0x00000100

/* XXX: improve that by looking at several start codes */
static int mpegvideo_probe(AVProbeData *p)
{
187 188
    int code;
    const uint8_t *d;
F
Fabrice Bellard 已提交
189 190 191 192

    /* we search the first start code. If it is a sequence, gop or
       picture start code then we decide it is an mpeg video
       stream. We do not send highest value to give a chance to mpegts */
193 194 195 196 197 198 199 200 201 202 203 204 205 206
    /* NOTE: the search range was restricted to avoid too many false
       detections */

    if (p->buf_size < 6)
        return 0;
    d = p->buf;
    code = (d[0] << 24) | (d[1] << 16) | (d[2] << 8) | (d[3]);
    if ((code & 0xffffff00) == 0x100) {
        if (code == SEQ_START_CODE ||
            code == GOP_START_CODE ||
            code == PICTURE_START_CODE)
            return 50 - 1;
        else
            return 0;
F
Fabrice Bellard 已提交
207 208 209 210 211 212
    }
    return 0;
}

AVInputFormat mp3_iformat = {
    "mp3",
F
Fabrice Bellard 已提交
213
    "MPEG audio",
F
Fabrice Bellard 已提交
214 215 216 217 218
    0,
    NULL,
    mp3_read_header,
    raw_read_packet,
    raw_read_close,
219
    .extensions = "mp2,mp3", /* XXX: use probe */
F
Fabrice Bellard 已提交
220 221 222 223 224
};

AVOutputFormat mp2_oformat = {
    "mp2",
    "MPEG audio layer 2",
F
Fabrice Bellard 已提交
225
    "audio/x-mpeg",
226 227 228
#ifdef CONFIG_MP3LAME
    "mp2",
#else
F
Fabrice Bellard 已提交
229
    "mp2,mp3",
230
#endif
F
Fabrice Bellard 已提交
231
    0,
F
Fabrice Bellard 已提交
232 233 234 235 236
    CODEC_ID_MP2,
    0,
    raw_write_header,
    raw_write_packet,
    raw_write_trailer,
F
Fabrice Bellard 已提交
237
};
F
Fabrice Bellard 已提交
238

239 240 241 242 243 244 245 246 247 248 249 250 251 252
#ifdef CONFIG_MP3LAME
AVOutputFormat mp3_oformat = {
    "mp3",
    "MPEG audio layer 3",
    "audio/x-mpeg",
    "mp3",
    0,
    CODEC_ID_MP3LAME,
    0,
    raw_write_header,
    raw_write_packet,
    raw_write_trailer,
};
#endif
F
Fabrice Bellard 已提交
253 254 255 256 257 258

AVInputFormat ac3_iformat = {
    "ac3",
    "raw ac3",
    0,
    NULL,
259
    ac3_read_header,
F
Fabrice Bellard 已提交
260 261
    raw_read_packet,
    raw_read_close,
262
    .extensions = "ac3",
F
Fabrice Bellard 已提交
263 264
};

F
Fabrice Bellard 已提交
265
AVOutputFormat ac3_oformat = {
F
Fabrice Bellard 已提交
266 267 268 269
    "ac3",
    "raw ac3",
    "audio/x-ac3", 
    "ac3",
F
Fabrice Bellard 已提交
270
    0,
F
Fabrice Bellard 已提交
271 272 273 274 275 276 277
    CODEC_ID_AC3,
    0,
    raw_write_header,
    raw_write_packet,
    raw_write_trailer,
};

F
Fabrice Bellard 已提交
278
AVOutputFormat h263_oformat = {
F
Fabrice Bellard 已提交
279 280 281 282 283
    "h263",
    "raw h263",
    "video/x-h263",
    "h263",
    0,
F
Fabrice Bellard 已提交
284
    0,
F
Fabrice Bellard 已提交
285 286 287 288
    CODEC_ID_H263,
    raw_write_header,
    raw_write_packet,
    raw_write_trailer,
F
Fabrice Bellard 已提交
289 290
};

M
Michael Niedermayer 已提交
291 292 293 294
AVInputFormat m4v_iformat = {
    "m4v",
    "raw MPEG4 video format",
    0,
295
    NULL /*mpegvideo_probe*/,
M
Michael Niedermayer 已提交
296 297 298
    video_read_header,
    raw_read_packet,
    raw_read_close,
299 300
    .extensions = "m4v", //FIXME remove after writing mpeg4_probe
    .value = CODEC_ID_MPEG4,
M
Michael Niedermayer 已提交
301 302
};

303 304 305 306 307 308 309 310 311 312 313 314 315
AVOutputFormat m4v_oformat = {
    "m4v",
    "raw MPEG4 video format",
    NULL,
    "m4v",
    0,
    CODEC_ID_NONE,
    CODEC_ID_MPEG4,
    raw_write_header,
    raw_write_packet,
    raw_write_trailer,
};

M
Michael Niedermayer 已提交
316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340
AVInputFormat h264_iformat = {
    "h264",
    "raw H264 video format",
    0,
    NULL /*mpegvideo_probe*/,
    video_read_header,
    raw_read_packet,
    raw_read_close,
    .extensions = "h26l,h264", //FIXME remove after writing mpeg4_probe
    .value = CODEC_ID_H264,
};

AVOutputFormat h264_oformat = {
    "h264",
    "raw H264 video format",
    NULL,
    "h264",
    0,
    CODEC_ID_NONE,
    CODEC_ID_H264,
    raw_write_header,
    raw_write_packet,
    raw_write_trailer,
};

F
Fabrice Bellard 已提交
341 342 343 344 345
AVInputFormat mpegvideo_iformat = {
    "mpegvideo",
    "MPEG video",
    0,
    mpegvideo_probe,
F
Fabrice Bellard 已提交
346 347 348
    video_read_header,
    raw_read_packet,
    raw_read_close,
349
    .value = CODEC_ID_MPEG1VIDEO,
F
Fabrice Bellard 已提交
350 351
};

F
Fabrice Bellard 已提交
352 353
AVOutputFormat mpeg1video_oformat = {
    "mpeg1video",
F
Fabrice Bellard 已提交
354 355 356 357
    "MPEG video",
    "video/x-mpeg",
    "mpg,mpeg",
    0,
F
Fabrice Bellard 已提交
358
    0,
F
Fabrice Bellard 已提交
359 360 361 362 363 364
    CODEC_ID_MPEG1VIDEO,
    raw_write_header,
    raw_write_packet,
    raw_write_trailer,
};

F
Fabrice Bellard 已提交
365
AVInputFormat mjpeg_iformat = {
F
Fabrice Bellard 已提交
366 367 368
    "mjpeg",
    "MJPEG video",
    0,
F
Fabrice Bellard 已提交
369
    NULL,
F
Fabrice Bellard 已提交
370 371 372
    video_read_header,
    raw_read_packet,
    raw_read_close,
373 374
    .extensions = "mjpg,mjpeg",
    .value = CODEC_ID_MJPEG,
F
Fabrice Bellard 已提交
375 376
};

F
Fabrice Bellard 已提交
377 378 379 380 381
AVOutputFormat mjpeg_oformat = {
    "mjpeg",
    "MJPEG video",
    "video/x-mjpeg",
    "mjpg,mjpeg",
F
Fabrice Bellard 已提交
382 383
    0,
    0,
F
Fabrice Bellard 已提交
384
    CODEC_ID_MJPEG,
F
Fabrice Bellard 已提交
385 386 387 388 389
    raw_write_header,
    raw_write_packet,
    raw_write_trailer,
};

F
Fabrice Bellard 已提交
390
/* pcm formats */
F
Fabrice Bellard 已提交
391

F
Fabrice Bellard 已提交
392 393 394 395 396 397 398 399 400
#define PCMDEF(name, long_name, ext, codec) \
AVInputFormat pcm_ ## name ## _iformat = {\
    #name,\
    long_name,\
    0,\
    NULL,\
    raw_read_header,\
    raw_read_packet,\
    raw_read_close,\
401 402
    .extensions = ext,\
    .value = codec,\
F
Fabrice Bellard 已提交
403 404 405 406 407 408 409 410 411 412 413 414 415
};\
\
AVOutputFormat pcm_ ## name ## _oformat = {\
    #name,\
    long_name,\
    NULL,\
    ext,\
    0,\
    codec,\
    0,\
    raw_write_header,\
    raw_write_packet,\
    raw_write_trailer,\
F
Fabrice Bellard 已提交
416 417 418
};

#ifdef WORDS_BIGENDIAN
F
Fabrice Bellard 已提交
419 420
#define BE_DEF(s) s
#define LE_DEF(s) NULL
F
Fabrice Bellard 已提交
421
#else
F
Fabrice Bellard 已提交
422 423
#define BE_DEF(s) NULL
#define LE_DEF(s) s
F
Fabrice Bellard 已提交
424 425 426
#endif


F
Fabrice Bellard 已提交
427 428
PCMDEF(s16le, "pcm signed 16 bit little endian format", 
       LE_DEF("sw"), CODEC_ID_PCM_S16LE)
F
Fabrice Bellard 已提交
429

F
Fabrice Bellard 已提交
430 431
PCMDEF(s16be, "pcm signed 16 bit big endian format", 
       BE_DEF("sw"), CODEC_ID_PCM_S16BE)
F
Fabrice Bellard 已提交
432

F
Fabrice Bellard 已提交
433 434
PCMDEF(u16le, "pcm unsigned 16 bit little endian format", 
       LE_DEF("uw"), CODEC_ID_PCM_U16LE)
F
Fabrice Bellard 已提交
435

F
Fabrice Bellard 已提交
436 437
PCMDEF(u16be, "pcm unsigned 16 bit big endian format", 
       BE_DEF("uw"), CODEC_ID_PCM_U16BE)
F
Fabrice Bellard 已提交
438

F
Fabrice Bellard 已提交
439 440
PCMDEF(s8, "pcm signed 8 bit format", 
       "sb", CODEC_ID_PCM_S8)
F
Fabrice Bellard 已提交
441

F
Fabrice Bellard 已提交
442 443
PCMDEF(u8, "pcm unsigned 8 bit format", 
       "ub", CODEC_ID_PCM_U8)
F
Fabrice Bellard 已提交
444

F
Fabrice Bellard 已提交
445 446
PCMDEF(mulaw, "pcm mu law format", 
       "ul", CODEC_ID_PCM_MULAW)
F
Fabrice Bellard 已提交
447

F
Fabrice Bellard 已提交
448 449
PCMDEF(alaw, "pcm A law format", 
       "al", CODEC_ID_PCM_ALAW)
F
Fabrice Bellard 已提交
450

451
static int rawvideo_read_packet(AVFormatContext *s, AVPacket *pkt)
F
Fabrice Bellard 已提交
452 453 454 455 456 457 458
{
    int packet_size, ret, width, height;
    AVStream *st = s->streams[0];

    width = st->codec.width;
    height = st->codec.height;

459 460
    packet_size = avpicture_get_size(st->codec.pix_fmt, width, height);
    if (packet_size < 0)
461
        av_abort();
F
Fabrice Bellard 已提交
462 463 464 465 466

    if (av_new_packet(pkt, packet_size) < 0)
        return -EIO;

    pkt->stream_index = 0;
M
Michael Niedermayer 已提交
467
#if 0
F
Fabrice Bellard 已提交
468 469
    /* bypass buffered I/O */
    ret = url_read(url_fileno(&s->pb), pkt->data, pkt->size);
M
Michael Niedermayer 已提交
470 471 472
#else
    ret = get_buffer(&s->pb, pkt->data, pkt->size);
#endif
F
Fabrice Bellard 已提交
473 474 475 476 477 478 479 480
    if (ret != pkt->size) {
        av_free_packet(pkt);
        return -EIO;
    } else {
        return 0;
    }
}

F
Fabrice Bellard 已提交
481 482 483 484 485 486 487 488
AVInputFormat rawvideo_iformat = {
    "rawvideo",
    "raw video format",
    0,
    NULL,
    raw_read_header,
    rawvideo_read_packet,
    raw_read_close,
489 490
    .extensions = "yuv",
    .value = CODEC_ID_RAWVIDEO,
F
Fabrice Bellard 已提交
491 492 493
};

AVOutputFormat rawvideo_oformat = {
F
Fabrice Bellard 已提交
494 495 496 497
    "rawvideo",
    "raw video format",
    NULL,
    "yuv",
F
Fabrice Bellard 已提交
498
    0,
F
Fabrice Bellard 已提交
499 500 501 502 503 504
    CODEC_ID_NONE,
    CODEC_ID_RAWVIDEO,
    raw_write_header,
    raw_write_packet,
    raw_write_trailer,
};
F
Fabrice Bellard 已提交
505

506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527
static int null_write_packet(struct AVFormatContext *s, 
                             int stream_index,
                             unsigned char *buf, int size, int force_pts)
{
    return 0;
}

AVOutputFormat null_oformat = {
    "null",
    "null video format",
    NULL,
    NULL,
    0,
#ifdef WORDS_BIGENDIAN
    CODEC_ID_PCM_S16BE,
#else
    CODEC_ID_PCM_S16LE,
#endif
    CODEC_ID_RAWVIDEO,
    raw_write_header,
    null_write_packet,
    raw_write_trailer,
528
    .flags = AVFMT_NOFILE | AVFMT_RAWPICTURE,
529 530
};

F
Fabrice Bellard 已提交
531 532 533 534
int raw_init(void)
{
    av_register_input_format(&mp3_iformat);
    av_register_output_format(&mp2_oformat);
535 536 537
#ifdef CONFIG_MP3LAME
    av_register_output_format(&mp3_oformat);
#endif    
F
Fabrice Bellard 已提交
538 539 540 541
    av_register_input_format(&ac3_iformat);
    av_register_output_format(&ac3_oformat);

    av_register_output_format(&h263_oformat);
542
    
M
Michael Niedermayer 已提交
543
    av_register_input_format(&m4v_iformat);
544
    av_register_output_format(&m4v_oformat);
M
Michael Niedermayer 已提交
545 546 547
    
    av_register_input_format(&h264_iformat);
    av_register_output_format(&h264_oformat);
F
Fabrice Bellard 已提交
548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573

    av_register_input_format(&mpegvideo_iformat);
    av_register_output_format(&mpeg1video_oformat);

    av_register_input_format(&mjpeg_iformat);
    av_register_output_format(&mjpeg_oformat);

    av_register_input_format(&pcm_s16le_iformat);
    av_register_output_format(&pcm_s16le_oformat);
    av_register_input_format(&pcm_s16be_iformat);
    av_register_output_format(&pcm_s16be_oformat);
    av_register_input_format(&pcm_u16le_iformat);
    av_register_output_format(&pcm_u16le_oformat);
    av_register_input_format(&pcm_u16be_iformat);
    av_register_output_format(&pcm_u16be_oformat);
    av_register_input_format(&pcm_s8_iformat);
    av_register_output_format(&pcm_s8_oformat);
    av_register_input_format(&pcm_u8_iformat);
    av_register_output_format(&pcm_u8_oformat);
    av_register_input_format(&pcm_mulaw_iformat);
    av_register_output_format(&pcm_mulaw_oformat);
    av_register_input_format(&pcm_alaw_iformat);
    av_register_output_format(&pcm_alaw_oformat);

    av_register_input_format(&rawvideo_iformat);
    av_register_output_format(&rawvideo_oformat);
574 575

    av_register_output_format(&null_oformat);
F
Fabrice Bellard 已提交
576 577
    return 0;
}