2000-08-28 03:18:25 +08:00
|
|
|
/** \ingroup rpmbuild
|
|
|
|
* \file build/build.c
|
2000-01-25 04:02:32 +08:00
|
|
|
* Top-level build dispatcher.
|
|
|
|
*/
|
|
|
|
|
1998-07-26 05:00:26 +08:00
|
|
|
#include "system.h"
|
1995-12-14 23:56:10 +08:00
|
|
|
|
2000-06-10 02:57:23 +08:00
|
|
|
#include <rpmio_internal.h>
|
2000-06-20 23:54:48 +08:00
|
|
|
#include <rpmbuild.h>
|
1995-12-14 23:56:10 +08:00
|
|
|
|
2000-12-13 04:03:45 +08:00
|
|
|
#include "debug.h"
|
|
|
|
|
2001-10-15 11:22:10 +08:00
|
|
|
/*@unchecked@*/
|
1999-11-24 08:03:54 +08:00
|
|
|
static int _build_debug = 0;
|
|
|
|
|
2001-03-15 07:09:09 +08:00
|
|
|
/*@access StringBuf @*/
|
2001-05-01 06:32:22 +08:00
|
|
|
/*@access urlinfo @*/ /* XXX compared with NULL */
|
2001-03-15 07:09:09 +08:00
|
|
|
/*@access FD_t @*/
|
|
|
|
|
2001-01-11 22:13:04 +08:00
|
|
|
/**
|
|
|
|
*/
|
1998-07-26 05:00:26 +08:00
|
|
|
static void doRmSource(Spec spec)
|
2001-10-15 11:22:10 +08:00
|
|
|
/*@globals rpmGlobalMacroContext,
|
|
|
|
fileSystem@*/
|
2001-10-18 00:43:36 +08:00
|
|
|
/*@modifies rpmGlobalMacroContext, fileSystem @*/
|
1995-11-28 06:31:21 +08:00
|
|
|
{
|
1998-07-26 05:00:26 +08:00
|
|
|
struct Source *p;
|
|
|
|
Package pkg;
|
2001-10-15 11:22:10 +08:00
|
|
|
int rc;
|
1998-07-26 05:00:26 +08:00
|
|
|
|
1999-04-30 23:55:45 +08:00
|
|
|
#if 0
|
2001-10-15 11:22:10 +08:00
|
|
|
rc = Unlink(spec->specFile);
|
1999-04-30 23:55:45 +08:00
|
|
|
#endif
|
1995-12-18 22:53:14 +08:00
|
|
|
|
1998-11-08 08:15:33 +08:00
|
|
|
for (p = spec->sources; p != NULL; p = p->next) {
|
1998-07-26 05:00:26 +08:00
|
|
|
if (! (p->flags & RPMBUILD_ISNO)) {
|
1999-01-06 07:13:56 +08:00
|
|
|
const char *fn = rpmGetPath("%{_sourcedir}/", p->source, NULL);
|
2001-10-15 11:22:10 +08:00
|
|
|
rc = Unlink(fn);
|
2001-04-29 09:05:43 +08:00
|
|
|
fn = _free(fn);
|
1995-12-28 00:54:54 +08:00
|
|
|
}
|
1998-07-26 05:00:26 +08:00
|
|
|
}
|
1995-12-28 00:54:54 +08:00
|
|
|
|
1998-11-08 08:15:33 +08:00
|
|
|
for (pkg = spec->packages; pkg != NULL; pkg = pkg->next) {
|
|
|
|
for (p = pkg->icon; p != NULL; p = p->next) {
|
1998-07-26 05:00:26 +08:00
|
|
|
if (! (p->flags & RPMBUILD_ISNO)) {
|
1999-01-06 07:13:56 +08:00
|
|
|
const char *fn = rpmGetPath("%{_sourcedir}/", p->source, NULL);
|
2001-10-15 11:22:10 +08:00
|
|
|
rc = Unlink(fn);
|
2001-04-29 09:05:43 +08:00
|
|
|
fn = _free(fn);
|
1998-01-13 05:31:29 +08:00
|
|
|
}
|
1995-12-28 00:54:54 +08:00
|
|
|
}
|
1998-07-26 05:00:26 +08:00
|
|
|
}
|
|
|
|
}
|
1995-12-28 00:54:54 +08:00
|
|
|
|
1998-09-06 04:02:08 +08:00
|
|
|
/*
|
2001-01-11 22:13:04 +08:00
|
|
|
* @todo Single use by %%doc in files.c prevents static.
|
1998-09-06 04:02:08 +08:00
|
|
|
*/
|
1999-01-07 01:33:50 +08:00
|
|
|
int doScript(Spec spec, int what, const char *name, StringBuf sb, int test)
|
1996-01-08 15:06:16 +08:00
|
|
|
{
|
1999-11-20 02:19:41 +08:00
|
|
|
const char * rootURL = spec->rootURL;
|
|
|
|
const char * rootDir;
|
|
|
|
const char *scriptName = NULL;
|
1999-11-24 08:03:54 +08:00
|
|
|
const char * buildDirURL = rpmGenPath(rootURL, "%{_builddir}", "");
|
1999-11-20 02:19:41 +08:00
|
|
|
const char * buildScript;
|
|
|
|
const char * buildCmd = NULL;
|
|
|
|
const char * buildTemplate = NULL;
|
|
|
|
const char * buildPost = NULL;
|
|
|
|
const char * mTemplate = NULL;
|
|
|
|
const char * mPost = NULL;
|
|
|
|
int argc = 0;
|
|
|
|
const char **argv = NULL;
|
|
|
|
FILE * fp = NULL;
|
|
|
|
urlinfo u = NULL;
|
|
|
|
|
1998-11-19 05:41:05 +08:00
|
|
|
FD_t fd;
|
1999-11-05 05:26:08 +08:00
|
|
|
FD_t xfd;
|
1999-11-20 02:19:41 +08:00
|
|
|
int child;
|
1998-01-13 05:31:29 +08:00
|
|
|
int status;
|
1999-11-20 02:19:41 +08:00
|
|
|
int rc;
|
1998-01-13 05:31:29 +08:00
|
|
|
|
2001-10-16 01:53:34 +08:00
|
|
|
/*@-branchstate@*/
|
1998-01-13 05:31:29 +08:00
|
|
|
switch (what) {
|
2001-10-16 01:53:34 +08:00
|
|
|
case RPMBUILD_PREP:
|
1998-01-13 05:31:29 +08:00
|
|
|
name = "%prep";
|
|
|
|
sb = spec->prep;
|
1999-11-20 02:19:41 +08:00
|
|
|
mTemplate = "%{__spec_prep_template}";
|
|
|
|
mPost = "%{__spec_prep_post}";
|
1998-01-13 05:31:29 +08:00
|
|
|
break;
|
2001-10-16 01:53:34 +08:00
|
|
|
case RPMBUILD_BUILD:
|
1998-01-13 05:31:29 +08:00
|
|
|
name = "%build";
|
|
|
|
sb = spec->build;
|
1999-11-20 02:19:41 +08:00
|
|
|
mTemplate = "%{__spec_build_template}";
|
|
|
|
mPost = "%{__spec_build_post}";
|
1998-01-13 05:31:29 +08:00
|
|
|
break;
|
2001-10-16 01:53:34 +08:00
|
|
|
case RPMBUILD_INSTALL:
|
1998-01-13 05:31:29 +08:00
|
|
|
name = "%install";
|
|
|
|
sb = spec->install;
|
1999-11-20 02:19:41 +08:00
|
|
|
mTemplate = "%{__spec_install_template}";
|
|
|
|
mPost = "%{__spec_install_post}";
|
1998-01-13 05:31:29 +08:00
|
|
|
break;
|
2001-10-16 01:53:34 +08:00
|
|
|
case RPMBUILD_CLEAN:
|
1998-01-13 05:31:29 +08:00
|
|
|
name = "%clean";
|
|
|
|
sb = spec->clean;
|
1999-11-20 02:19:41 +08:00
|
|
|
mTemplate = "%{__spec_clean_template}";
|
|
|
|
mPost = "%{__spec_clean_post}";
|
1998-01-13 05:31:29 +08:00
|
|
|
break;
|
2001-10-16 01:53:34 +08:00
|
|
|
case RPMBUILD_RMBUILD:
|
1998-01-13 05:31:29 +08:00
|
|
|
name = "--clean";
|
1999-11-20 02:19:41 +08:00
|
|
|
mTemplate = "%{__spec_clean_template}";
|
|
|
|
mPost = "%{__spec_clean_post}";
|
1998-01-13 05:31:29 +08:00
|
|
|
break;
|
2001-10-16 01:53:34 +08:00
|
|
|
case RPMBUILD_STRINGBUF:
|
|
|
|
default:
|
1999-11-20 02:19:41 +08:00
|
|
|
mTemplate = "%{___build_template}";
|
|
|
|
mPost = "%{___build_post}";
|
1998-01-13 05:31:29 +08:00
|
|
|
break;
|
|
|
|
}
|
2001-10-16 01:53:34 +08:00
|
|
|
/*@=branchstate@*/
|
1999-11-01 05:38:21 +08:00
|
|
|
|
1999-11-20 02:19:41 +08:00
|
|
|
if ((what != RPMBUILD_RMBUILD) && sb == NULL) {
|
|
|
|
rc = 0;
|
|
|
|
goto exit;
|
|
|
|
}
|
1998-01-13 05:31:29 +08:00
|
|
|
|
1999-11-24 08:03:54 +08:00
|
|
|
if (makeTempFile(rootURL, &scriptName, &fd) || fd == NULL || Ferror(fd)) {
|
2001-01-16 07:09:42 +08:00
|
|
|
rpmError(RPMERR_SCRIPT, _("Unable to open temp file.\n"));
|
1999-11-20 02:19:41 +08:00
|
|
|
rc = RPMERR_SCRIPT;
|
|
|
|
goto exit;
|
1996-01-08 15:06:16 +08:00
|
|
|
}
|
1999-11-20 02:19:41 +08:00
|
|
|
|
1999-06-18 05:35:34 +08:00
|
|
|
#ifdef HAVE_FCHMOD
|
1999-11-20 02:19:41 +08:00
|
|
|
switch (rootut) {
|
|
|
|
case URL_IS_PATH:
|
|
|
|
case URL_IS_UNKNOWN:
|
1999-11-24 08:03:54 +08:00
|
|
|
(void)fchmod(Fileno(fd), 0600);
|
1999-11-20 02:19:41 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
1999-11-15 03:15:18 +08:00
|
|
|
#endif
|
1999-11-20 02:19:41 +08:00
|
|
|
|
2001-10-16 01:53:34 +08:00
|
|
|
/*@-branchstate@*/
|
1999-11-20 02:19:41 +08:00
|
|
|
if (fdGetFp(fd) == NULL)
|
|
|
|
xfd = Fdopen(fd, "w.fpio");
|
|
|
|
else
|
|
|
|
xfd = fd;
|
2001-10-16 01:53:34 +08:00
|
|
|
/*@=branchstate@*/
|
|
|
|
|
2001-10-17 01:42:18 +08:00
|
|
|
/*@-type@*/ /* FIX: cast? */
|
1999-11-20 02:19:41 +08:00
|
|
|
if ((fp = fdGetFp(xfd)) == NULL) {
|
|
|
|
rc = RPMERR_SCRIPT;
|
|
|
|
goto exit;
|
|
|
|
}
|
2001-10-17 01:42:18 +08:00
|
|
|
/*@=type@*/
|
1995-12-28 00:54:54 +08:00
|
|
|
|
1999-11-20 02:19:41 +08:00
|
|
|
(void) urlPath(rootURL, &rootDir);
|
2001-10-16 01:53:34 +08:00
|
|
|
/*@-branchstate@*/
|
1999-11-20 02:19:41 +08:00
|
|
|
if (*rootDir == '\0') rootDir = "/";
|
2001-10-16 01:53:34 +08:00
|
|
|
/*@=branchstate@*/
|
1998-09-06 07:13:35 +08:00
|
|
|
|
1999-11-20 02:19:41 +08:00
|
|
|
(void) urlPath(scriptName, &buildScript);
|
1998-09-06 04:02:08 +08:00
|
|
|
|
1999-11-20 02:19:41 +08:00
|
|
|
buildTemplate = rpmExpand(mTemplate, NULL);
|
|
|
|
buildPost = rpmExpand(mPost, NULL);
|
|
|
|
|
2001-05-01 06:32:22 +08:00
|
|
|
(void) fputs(buildTemplate, fp);
|
1999-11-20 02:19:41 +08:00
|
|
|
|
|
|
|
if (what != RPMBUILD_PREP && what != RPMBUILD_RMBUILD && spec->buildSubdir)
|
|
|
|
fprintf(fp, "cd %s\n", spec->buildSubdir);
|
1998-09-06 04:02:08 +08:00
|
|
|
|
1998-01-13 05:31:29 +08:00
|
|
|
if (what == RPMBUILD_RMBUILD) {
|
1999-11-01 05:38:21 +08:00
|
|
|
if (spec->buildSubdir)
|
1999-11-19 02:07:46 +08:00
|
|
|
fprintf(fp, "rm -rf %s\n", spec->buildSubdir);
|
1999-11-01 05:38:21 +08:00
|
|
|
} else
|
1999-11-19 02:07:46 +08:00
|
|
|
fprintf(fp, "%s", getStringBuf(sb));
|
1999-11-20 02:19:41 +08:00
|
|
|
|
2001-05-01 06:32:22 +08:00
|
|
|
(void) fputs(buildPost, fp);
|
1998-01-13 05:31:29 +08:00
|
|
|
|
2001-05-01 06:32:22 +08:00
|
|
|
(void) Fclose(xfd);
|
1995-12-28 00:54:54 +08:00
|
|
|
|
1998-01-13 05:31:29 +08:00
|
|
|
if (test) {
|
1999-11-20 02:19:41 +08:00
|
|
|
rc = 0;
|
|
|
|
goto exit;
|
1995-12-28 00:54:54 +08:00
|
|
|
}
|
1998-01-13 05:31:29 +08:00
|
|
|
|
1999-11-24 08:03:54 +08:00
|
|
|
if (_build_debug)
|
|
|
|
fprintf(stderr, "*** rootURL %s buildDirURL %s\n", rootURL, buildDirURL);
|
|
|
|
if (buildDirURL && buildDirURL[0] != '/' &&
|
|
|
|
(urlSplit(buildDirURL, &u) != 0)) {
|
1999-11-20 02:19:41 +08:00
|
|
|
rc = RPMERR_SCRIPT;
|
|
|
|
goto exit;
|
|
|
|
}
|
2001-05-01 06:32:22 +08:00
|
|
|
if (u != NULL) {
|
1999-11-24 08:03:54 +08:00
|
|
|
switch (u->urltype) {
|
|
|
|
case URL_IS_FTP:
|
|
|
|
if (_build_debug)
|
|
|
|
fprintf(stderr, "*** addMacros\n");
|
|
|
|
addMacro(spec->macros, "_remsh", NULL, "%{__remsh}", RMIL_SPEC);
|
|
|
|
addMacro(spec->macros, "_remhost", NULL, u->host, RMIL_SPEC);
|
|
|
|
if (strcmp(rootDir, "/"))
|
|
|
|
addMacro(spec->macros, "_remroot", NULL, rootDir, RMIL_SPEC);
|
|
|
|
break;
|
|
|
|
case URL_IS_HTTP:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
1999-11-20 02:19:41 +08:00
|
|
|
|
|
|
|
buildCmd = rpmExpand("%{___build_cmd}", " ", buildScript, NULL);
|
2001-05-01 06:32:22 +08:00
|
|
|
(void) poptParseArgvString(buildCmd, &argc, &argv);
|
1999-11-20 02:19:41 +08:00
|
|
|
|
|
|
|
rpmMessage(RPMMESS_NORMAL, _("Executing(%s): %s\n"), name, buildCmd);
|
|
|
|
if (!(child = fork())) {
|
|
|
|
|
2001-10-18 00:43:36 +08:00
|
|
|
/*@-mods@*/
|
1999-11-24 08:03:54 +08:00
|
|
|
errno = 0;
|
2001-10-18 00:43:36 +08:00
|
|
|
/*@=mods@*/
|
2001-05-01 06:32:22 +08:00
|
|
|
(void) execvp(argv[0], (char *const *)argv);
|
1999-11-20 02:19:41 +08:00
|
|
|
|
2001-01-16 07:09:42 +08:00
|
|
|
rpmError(RPMERR_SCRIPT, _("Exec of %s failed (%s): %s\n"),
|
|
|
|
scriptName, name, strerror(errno));
|
1999-11-01 05:38:21 +08:00
|
|
|
|
|
|
|
_exit(-1);
|
1996-01-29 11:32:22 +08:00
|
|
|
}
|
1999-11-01 05:38:21 +08:00
|
|
|
|
1999-11-20 02:19:41 +08:00
|
|
|
rc = waitpid(child, &status, 0);
|
|
|
|
|
|
|
|
if (!WIFEXITED(status) || WEXITSTATUS(status)) {
|
2001-01-16 07:09:42 +08:00
|
|
|
rpmError(RPMERR_SCRIPT, _("Bad exit status from %s (%s)\n"),
|
1998-01-13 05:31:29 +08:00
|
|
|
scriptName, name);
|
1999-11-20 02:19:41 +08:00
|
|
|
rc = RPMERR_SCRIPT;
|
|
|
|
} else
|
|
|
|
rc = 0;
|
1998-01-13 05:31:29 +08:00
|
|
|
|
1999-11-20 02:19:41 +08:00
|
|
|
exit:
|
|
|
|
if (scriptName) {
|
|
|
|
if (!rc)
|
2001-05-01 06:32:22 +08:00
|
|
|
(void) Unlink(scriptName);
|
2001-04-29 09:05:43 +08:00
|
|
|
scriptName = _free(scriptName);
|
1999-11-20 02:19:41 +08:00
|
|
|
}
|
2001-05-01 06:32:22 +08:00
|
|
|
if (u != NULL) {
|
1999-11-24 08:03:54 +08:00
|
|
|
switch (u->urltype) {
|
|
|
|
case URL_IS_FTP:
|
|
|
|
case URL_IS_HTTP:
|
|
|
|
if (_build_debug)
|
|
|
|
fprintf(stderr, "*** delMacros\n");
|
|
|
|
delMacro(spec->macros, "_remsh");
|
|
|
|
delMacro(spec->macros, "_remhost");
|
|
|
|
if (strcmp(rootDir, "/"))
|
|
|
|
delMacro(spec->macros, "_remroot");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2001-04-29 09:05:43 +08:00
|
|
|
argv = _free(argv);
|
|
|
|
buildCmd = _free(buildCmd);
|
|
|
|
buildTemplate = _free(buildTemplate);
|
|
|
|
buildPost = _free(buildPost);
|
|
|
|
buildDirURL = _free(buildDirURL);
|
1995-12-14 23:56:10 +08:00
|
|
|
|
1999-11-20 02:19:41 +08:00
|
|
|
return rc;
|
1995-12-14 23:56:10 +08:00
|
|
|
}
|
|
|
|
|
1998-07-26 05:00:26 +08:00
|
|
|
int buildSpec(Spec spec, int what, int test)
|
1995-12-14 23:56:10 +08:00
|
|
|
{
|
2000-12-03 05:53:44 +08:00
|
|
|
int rc = 0;
|
1998-01-13 05:31:29 +08:00
|
|
|
|
2001-05-07 03:17:14 +08:00
|
|
|
if (!spec->recursing && spec->BACount) {
|
2000-12-03 05:53:44 +08:00
|
|
|
int x;
|
2001-05-07 03:17:14 +08:00
|
|
|
/* When iterating over BANames, do the source */
|
1998-07-26 05:00:26 +08:00
|
|
|
/* packaging on the first run, and skip RMSOURCE altogether */
|
2001-05-07 03:17:14 +08:00
|
|
|
if (spec->BASpecs != NULL)
|
|
|
|
for (x = 0; x < spec->BACount; x++) {
|
|
|
|
if ((rc = buildSpec(spec->BASpecs[x],
|
1998-07-26 05:00:26 +08:00
|
|
|
(what & ~RPMBUILD_RMSOURCE) |
|
|
|
|
(x ? 0 : (what & RPMBUILD_PACKAGESOURCE)),
|
|
|
|
test))) {
|
2000-09-02 05:15:40 +08:00
|
|
|
goto exit;
|
1998-07-26 05:00:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
1999-11-01 05:38:21 +08:00
|
|
|
if ((what & RPMBUILD_PREP) &&
|
|
|
|
(rc = doScript(spec, RPMBUILD_PREP, NULL, NULL, test)))
|
2000-09-02 05:15:40 +08:00
|
|
|
goto exit;
|
1999-11-01 05:38:21 +08:00
|
|
|
|
|
|
|
if ((what & RPMBUILD_BUILD) &&
|
|
|
|
(rc = doScript(spec, RPMBUILD_BUILD, NULL, NULL, test)))
|
2000-09-02 05:15:40 +08:00
|
|
|
goto exit;
|
1999-11-01 05:38:21 +08:00
|
|
|
|
|
|
|
if ((what & RPMBUILD_INSTALL) &&
|
|
|
|
(rc = doScript(spec, RPMBUILD_INSTALL, NULL, NULL, test)))
|
2000-09-02 05:15:40 +08:00
|
|
|
goto exit;
|
1995-12-20 05:08:18 +08:00
|
|
|
|
1999-11-01 05:38:21 +08:00
|
|
|
if ((what & RPMBUILD_PACKAGESOURCE) &&
|
|
|
|
(rc = processSourceFiles(spec)))
|
2000-09-02 05:15:40 +08:00
|
|
|
goto exit;
|
1995-12-20 05:08:18 +08:00
|
|
|
|
1999-11-01 05:38:21 +08:00
|
|
|
if (((what & RPMBUILD_INSTALL) || (what & RPMBUILD_PACKAGEBINARY) ||
|
|
|
|
(what & RPMBUILD_FILECHECK)) &&
|
|
|
|
(rc = processBinaryFiles(spec, what & RPMBUILD_INSTALL, test)))
|
2000-09-02 05:15:40 +08:00
|
|
|
goto exit;
|
1998-04-10 00:46:30 +08:00
|
|
|
|
1999-11-01 05:38:21 +08:00
|
|
|
if (((what & RPMBUILD_PACKAGESOURCE) && !test) &&
|
|
|
|
(rc = packageSources(spec)))
|
2000-12-03 05:53:44 +08:00
|
|
|
return rc;
|
1999-11-01 05:38:21 +08:00
|
|
|
|
|
|
|
if (((what & RPMBUILD_PACKAGEBINARY) && !test) &&
|
|
|
|
(rc = packageBinaries(spec)))
|
2000-09-02 05:15:40 +08:00
|
|
|
goto exit;
|
1998-07-26 05:00:26 +08:00
|
|
|
|
1999-11-01 05:38:21 +08:00
|
|
|
if ((what & RPMBUILD_CLEAN) &&
|
|
|
|
(rc = doScript(spec, RPMBUILD_CLEAN, NULL, NULL, test)))
|
2000-09-02 05:15:40 +08:00
|
|
|
goto exit;
|
1999-11-01 05:38:21 +08:00
|
|
|
|
|
|
|
if ((what & RPMBUILD_RMBUILD) &&
|
|
|
|
(rc = doScript(spec, RPMBUILD_RMBUILD, NULL, NULL, test)))
|
2000-09-02 05:15:40 +08:00
|
|
|
goto exit;
|
1998-04-10 00:46:30 +08:00
|
|
|
}
|
1998-07-26 05:00:26 +08:00
|
|
|
|
1999-11-01 05:38:21 +08:00
|
|
|
if (what & RPMBUILD_RMSOURCE)
|
1998-07-26 05:00:26 +08:00
|
|
|
doRmSource(spec);
|
|
|
|
|
1999-11-01 05:38:21 +08:00
|
|
|
if (what & RPMBUILD_RMSPEC)
|
2001-05-01 06:32:22 +08:00
|
|
|
(void) Unlink(spec->specFile);
|
1999-04-30 23:55:45 +08:00
|
|
|
|
2000-09-02 05:15:40 +08:00
|
|
|
exit:
|
|
|
|
if (rc && rpmlogGetNrecs() > 0) {
|
|
|
|
rpmMessage(RPMMESS_NORMAL, _("\n\nRPM build errors:\n"));
|
|
|
|
rpmlogPrint(NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
1995-12-20 05:08:18 +08:00
|
|
|
}
|