diff --git a/scheduler/Makefile b/scheduler/Makefile index 0decf8f..a35ee82 100644 --- a/scheduler/Makefile +++ b/scheduler/Makefile @@ -26,6 +26,7 @@ CUPSDOBJS = \ env.o \ main.o \ ipp.o \ + colord.o \ listen.o \ job.o \ log.o \ diff --git a/scheduler/colord.c b/scheduler/colord.c new file mode 100644 index 0000000..2fdf401 --- /dev/null +++ b/scheduler/colord.c @@ -0,0 +1,665 @@ +/* + * "$Id$" + * + * colord integration for the CUPS scheduler. + * + * Copyright 2011, Red Hat. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + * + * Contents: + * + * colordRegisterPrinter() - Register profiles for a printer. + * colordUnregisterPrinter() - Unregister profiles for a printer. + * colordStart() - Get a connection to the system bus. + * colordStop() - Release any connection to the system bus + * so that added profiles and devices are + * automatically removed. + */ + +/* + * Include necessary headers... + */ + +#include "cupsd.h" + +#ifdef HAVE_DBUS + +#include +#include + +/* + * Defines used by colord. See the reference docs for further details: + * http://colord.hughsie.com/api/ref-dbus.html + */ +#define COLORD_SCOPE_NORMAL "normal" /* System scope */ +#define COLORD_SCOPE_TEMP "temp" /* Process scope */ +#define COLORD_SCOPE_DISK "disk" /* Lives forever, as stored in DB */ + +#define COLORD_RELATION_SOFT "soft" /* Mapping is not default */ +#define COLORD_RELATION_HARD "hard" /* Explicitly mapped profile */ + +#define COLORD_SPACE_RGB "rgb" /* RGB colorspace */ +#define COLORD_SPACE_CMYK "cmyk" /* CMYK colorspace */ + +#define COLORD_MODE_PHYSICAL "physical" /* Actual device */ +#define COLORD_MODE_VIRTUAL "virtual" /* Virtual device with no hardware */ + +#define COLORD_KIND_PRINTER "printer" /* printing output device */ + +/* This is static */ +static DBusConnection *con = NULL; + +/* + * 'colordStart()' - Get a connection to the system bus. + */ + +void +colordStart(void) +{ + if (con) + return; + con = dbus_bus_get (DBUS_BUS_SYSTEM, NULL); +} + +/* + * 'colordStop()' - Release any connection to the system bus so that + * added profiles and devices are automatically removed. + */ + +void +colordStop(void) +{ + if (con == NULL) + return; + dbus_connection_unref(con); + con = NULL; +} + +/* + * 'message_dict_add_strings()' - add two strings to a dictionary. + */ + +static void +message_dict_add_strings (DBusMessageIter *dict, + const char *key, + const char *value) +{ + DBusMessageIter entry; + dbus_message_iter_open_container(dict, + DBUS_TYPE_DICT_ENTRY, + NULL, + &entry); + dbus_message_iter_append_basic(&entry, DBUS_TYPE_STRING, &key); + dbus_message_iter_append_basic(&entry, DBUS_TYPE_STRING, &value); + dbus_message_iter_close_container(dict, &entry); +} + +/* + * 'colordCreateProfile()' - Create a color profile for a printer. + * + * Notes: When creating the device, we can create + */ + +static void +colordCreateProfile (cups_array_t *profiles, /* I - Profiles array */ + const char *printer_name, /* I - Printer name */ + const char *qualifier, /* I - Profile qualifier */ + const char **format, /* I - Profile qualifier format */ + const char *iccfile, /* I - ICC filename */ + const char *scope) /* I - The scope of the profile, e.g. + 'normal', 'temp' or 'disk' */ +{ + DBusMessage *message = NULL; /* D-Bus request */ + DBusMessage *reply = NULL; /* D-Bus reply */ + DBusMessageIter args; /* D-Bus method arguments */ + DBusMessageIter dict; /* D-Bus method arguments */ + DBusError error; /* D-Bus error */ + int options = 1; /* Options for CreateDevice */ + char *idstr; /* Profile ID string */ + size_t idstrlen; /* Profile ID allocated length */ + const char *profile_path; /* Device object path */ + char format_str[1024]; /* Qualifier format as a string */ + + /* + * Create the profile... + */ + + message = dbus_message_new_method_call("org.freedesktop.ColorManager", + "/org/freedesktop/ColorManager", + "org.freedesktop.ColorManager", + "CreateProfile"); + + /* create a profile id */ + idstrlen = strlen (printer_name) + 1 + strlen (qualifier) + 1; + idstr = malloc (idstrlen); + if (!idstr) + goto out; + snprintf (idstr, idstrlen, "%s-%s", printer_name, qualifier); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Using profile id of %s", + idstr); + + dbus_message_iter_init_append(message, &args); + dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &idstr); + dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &scope); + + /* mush the qualifier format into a simple string */ + snprintf(format_str, sizeof(format_str), "%s.%s.%s", + format[0], + format[1], + format[2]); + + /* set initial properties */ + dbus_message_iter_open_container(&args, + DBUS_TYPE_ARRAY, + "{ss}", + &dict); + message_dict_add_strings(&dict, "Qualifier", qualifier); + message_dict_add_strings(&dict, "Format", format_str); + if (iccfile != NULL) + message_dict_add_strings(&dict, "Filename", iccfile); + dbus_message_iter_close_container(&args, &dict); + + /* send syncronous */ + dbus_error_init(&error); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Calling CreateProfile(%s,%d)", + idstr, options); + reply = dbus_connection_send_with_reply_and_block(con, + message, + -1, + &error); + if (reply == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "failed to CreateProfile: %s:%s", + error.name, error.message); + dbus_error_free(&error); + goto out; + } + + /* get reply data */ + dbus_message_iter_init(reply, &args); + if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_OBJECT_PATH) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "incorrect reply type"); + goto out; + } + dbus_message_iter_get_basic(&args, &profile_path); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "created profile %s", + profile_path); + cupsArrayAdd(profiles, strdup(profile_path)); + +out: + if (message != NULL) + dbus_message_unref(message); + if (reply != NULL) + dbus_message_unref(reply); + free (idstr); +} + +/* + * 'colordDeviceAddProfile()' - Assign a profile to a device. + */ + +static void +colordDeviceAddProfile (const char *device_path, /* I - Device object path */ + const char *profile_path, /* I - Profile object path */ + const char *relation) /* I - Device relation, either 'soft' or 'hard' */ +{ + DBusMessage *message = NULL; /* D-Bus request */ + DBusMessage *reply = NULL; /* D-Bus reply */ + DBusMessageIter args; /* D-Bus method arguments */ + DBusError error; /* D-Bus error */ + + message = dbus_message_new_method_call("org.freedesktop.ColorManager", + device_path, + "org.freedesktop.ColorManager.Device", + "AddProfile"); + + /* send profile path as the argument */ + dbus_message_iter_init_append(message, &args); + dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &relation); + dbus_message_iter_append_basic(&args, DBUS_TYPE_OBJECT_PATH, &profile_path); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "Calling %s:AddProfile(%s) [%s]", + device_path, profile_path, relation); + + /* send syncronous */ + dbus_error_init(&error); + reply = dbus_connection_send_with_reply_and_block(con, + message, + -1, + &error); + if (reply == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "failed to AddProfile: %s:%s", + error.name, error.message); + dbus_error_free(&error); + goto out; + } +out: + if (message != NULL) + dbus_message_unref(message); + if (reply != NULL) + dbus_message_unref(reply); +} + +/* + * 'colordCreateDevice()' - Create a device and register profiles. + */ + +static void +colordCreateDevice (cupsd_printer_t *p, /* I - Printer */ + cups_array_t *profiles, /* I - Profiles array */ + const char *colorspace, /* I - Device colorspace, e.g. 'rgb' */ + const char *relation, /* I - Profile relation, either 'soft' or 'hard' */ + const char *scope) /* I - The scope of the device, e.g. + 'normal', 'temp' or 'disk' */ +{ + DBusMessage *message = NULL; /* D-Bus request */ + DBusMessage *reply = NULL; /* D-Bus reply */ + DBusMessageIter args; /* D-Bus method arguments */ + DBusMessageIter dict; /* D-Bus method arguments */ + DBusError error; /* D-Bus error */ + const char *device_path; /* Device object path */ + const char *profile_path; /* Profile path */ + char *default_profile_path = NULL; + /* Default profile path */ + char device_id[1024]; /* Device ID as understood by colord */ + + /* + * Create the device... + */ + + snprintf(device_id, sizeof(device_id), "cups-%s", p->name); + device_path = device_id; + + message = dbus_message_new_method_call("org.freedesktop.ColorManager", + "/org/freedesktop/ColorManager", + "org.freedesktop.ColorManager", + "CreateDevice"); + + dbus_message_iter_init_append(message, &args); + dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &device_path); + dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &scope); + + /* set initial properties */ + dbus_message_iter_open_container(&args, + DBUS_TYPE_ARRAY, + "{ss}", + &dict); + message_dict_add_strings(&dict, "Colorspace", colorspace); + message_dict_add_strings(&dict, "Mode", COLORD_MODE_PHYSICAL); + if (p->make_model != NULL) + message_dict_add_strings(&dict, "Vendor", p->make_model); + message_dict_add_strings(&dict, "Model", p->name); + if (p->sanitized_device_uri != NULL) + message_dict_add_strings(&dict, "Serial", p->sanitized_device_uri); + message_dict_add_strings(&dict, "Kind", COLORD_KIND_PRINTER); + dbus_message_iter_close_container(&args, &dict); + + /* send syncronous */ + dbus_error_init(&error); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Calling CreateDevice(%s,%s)", + device_id, scope); + reply = dbus_connection_send_with_reply_and_block(con, + message, + -1, + &error); + if (reply == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "failed to CreateDevice: %s:%s", + error.name, error.message); + dbus_error_free(&error); + goto out; + } + + /* get reply data */ + dbus_message_iter_init(reply, &args); + if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_OBJECT_PATH) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "incorrect reply type"); + goto out; + } + dbus_message_iter_get_basic(&args, &device_path); + cupsdLogMessage(CUPSD_LOG_DEBUG, + "created device %s", + device_path); + + /* add profiles */ + for (profile_path = cupsArrayFirst(profiles); + profile_path; + profile_path = cupsArrayNext(profiles)) + { + colordDeviceAddProfile (device_path, profile_path, relation); + } + +out: + free(default_profile_path); + if (message != NULL) + dbus_message_unref(message); + if (reply != NULL) + dbus_message_unref(reply); +} + +/* + * 'colordDeleteDevice()' - Delete a device + */ + +static void +colordDeleteDevice (const char *device_id) /* I - Device ID string */ +{ + DBusMessage *message = NULL; /* D-Bus request */ + DBusMessage *reply = NULL; /* D-Bus reply */ + DBusMessageIter args; /* D-Bus method arguments */ + DBusError error; /* D-Bus error */ + + /* + * Create the device... + */ + + message = dbus_message_new_method_call("org.freedesktop.ColorManager", + "/org/freedesktop/ColorManager", + "org.freedesktop.ColorManager", + "DeleteDevice"); + + dbus_message_iter_init_append(message, &args); + dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &device_id); + + /* send syncronous */ + dbus_error_init(&error); + cupsdLogMessage(CUPSD_LOG_DEBUG, "Calling DeleteDevice(%s)", device_id); + reply = dbus_connection_send_with_reply_and_block(con, + message, + -1, + &error); + if (reply == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "failed to CreateDevice: %s:%s", + error.name, error.message); + dbus_error_free(&error); + goto out; + } +out: + if (message != NULL) + dbus_message_unref(message); + if (reply != NULL) + dbus_message_unref(reply); +} + +/* + * 'colordGetQualifierFormat()' - Get the qualifier format. + * + * Notes: Returns a value of "ColorSpace.MediaType.Resolution" by default + */ + +char ** +colordGetQualifierFormat(ppd_file_t *ppd) +{ + char **format; /* Qualifier format tuple */ + const char *tmp; /* Temporary string */ + ppd_attr_t *attr; /* Profile attributes */ + + /* create 3-tuple */ + format = calloc(3, sizeof(char*)); + + /* get 1st section */ + tmp = "cupsICCQualifier1"; + attr = ppdFindAttr(ppd, tmp, NULL); + if (attr != NULL) + tmp = attr->value; + else + { + tmp = "DefaultColorSpace"; + attr = ppdFindAttr(ppd, tmp, NULL); + } + if (attr == NULL) + { + tmp = "DefaultColorModel"; + attr = ppdFindAttr(ppd, tmp, NULL); + } + if (attr == NULL) + { + tmp = ""; + } + if (strncmp(tmp, "Default", 7) == 0) + tmp += 7; + format[0] = strdup(tmp); + + /* get 2st section */ + tmp = "cupsICCQualifier2"; + attr = ppdFindAttr(ppd, tmp, NULL); + if (attr != NULL) + tmp = attr->value; + else + { + tmp = "DefaultMediaType"; + attr = ppdFindAttr(ppd, tmp, NULL); + } + if (attr == NULL) + { + tmp = ""; + } + if (strncmp(tmp, "Default", 7) == 0) + tmp += 7; + format[1] = strdup(tmp); + + /* get 3st section */ + tmp = "cupsICCQualifier3"; + attr = ppdFindAttr(ppd, tmp, NULL); + if (attr != NULL) + tmp = attr->value; + else + { + tmp = "DefaultResolution"; + attr = ppdFindAttr(ppd, tmp, NULL); + } + if (attr == NULL) + { + tmp = ""; + } + if (strncmp(tmp, "Default", 7) == 0) + tmp += 7; + format[2] = strdup(tmp); + + return format; +} + +/* + * 'colordRegisterPrinter()' - Register profiles for a printer. + */ + +void +colordRegisterPrinter(cupsd_printer_t *p) /* I - printer */ +{ + char ppdfile[1024], /* PPD filename */ + iccfile[1024]; /* ICC filename */ + ppd_file_t *ppd; /* PPD file */ + char *profile_path; /* Profile path */ + cups_array_t *profiles; /* Profile paths array */ + const char *profile_key; /* Profile keyword */ + ppd_attr_t *attr; /* Profile attributes */ + const char *device_colorspace; /* Device colorspace */ + char **format; /* Qualifier format tuple */ + int i; /* Loop counter */ + + /* + * Ensure we have a DBus connection + */ + + colordStart(); + + /* + * Try opening the PPD file for this printer... + */ + + snprintf(ppdfile, sizeof(ppdfile), "%s/ppd/%s.ppd", ServerRoot, p->name); + if ((ppd = ppdOpenFile(ppdfile)) == NULL) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "cannot open %s", + ppdfile); + return; + } + + /* + * Find out the qualifier format + */ + + format = colordGetQualifierFormat(ppd); + + /* + * See if we have any embedded profiles... + */ + + profiles = cupsArrayNew (NULL, NULL); + profile_key = "APTiogaProfile"; + attr = ppdFindAttr(ppd, profile_key, NULL); + if (attr == NULL) + { + profile_key = "cupsICCProfile"; + attr = ppdFindAttr(ppd, profile_key, NULL); + } + for (; attr; attr = ppdFindNextAttr(ppd, profile_key, NULL)) + if (attr->spec[0] && attr->value && attr->value[0]) + { + if (attr->value[0] != '/') + snprintf(iccfile, sizeof(iccfile), "%s/profiles/%s", DataDir, + attr->value); + else + strlcpy(iccfile, attr->value, sizeof(iccfile)); + + if (access(iccfile, 0)) + { + cupsdLogMessage(CUPSD_LOG_WARN, + "no access to %s", + iccfile); + continue; + } + + colordCreateProfile(profiles, + p->name, + attr->spec, + (const char **)format, + iccfile, + COLORD_SCOPE_TEMP); + } + + /* + * Add the grayscale profile first. We always have a grayscale profile. + */ + + colordCreateProfile(profiles, + p->name, + "Gray..", + (const char **)format, + NULL, + COLORD_SCOPE_TEMP); + + /* + * Then add the RGB/CMYK/DeviceN color profile... + */ + + device_colorspace = "unknown"; + switch (ppd->colorspace) + { + case PPD_CS_RGB : + case PPD_CS_CMY : + device_colorspace = COLORD_SPACE_RGB; + colordCreateProfile(profiles, + p->name, + "RGB..", + (const char **)format, + NULL, + COLORD_SCOPE_TEMP); + break; + case PPD_CS_RGBK : + case PPD_CS_CMYK : + device_colorspace = COLORD_SPACE_CMYK; + colordCreateProfile(profiles, + p->name, + "CMYK..", + (const char **)format, + NULL, + COLORD_SCOPE_TEMP); + break; + case PPD_CS_GRAY : + if (attr) + break; + case PPD_CS_N : + colordCreateProfile(profiles, + p->name, + "DeviceN..", + (const char **)format, + NULL, + COLORD_SCOPE_TEMP); + break; + } + + /* + * Register the device with colord. + */ + + cupsdLogMessage(CUPSD_LOG_INFO, "Registering ICC color profiles for \"%s\"", + p->name); + colordCreateDevice (p, + profiles, + device_colorspace, + COLORD_RELATION_SOFT, + COLORD_SCOPE_TEMP); + + /* + * Free any memory we used... + */ + + for (profile_path = cupsArrayFirst(profiles); + profile_path != NULL; + profile_path = cupsArrayNext(profiles)) { + free(profile_path); + } + cupsArrayDelete(profiles); + for (i=0; i<3; i++) + free(format[i]); + free(format); + + ppdClose(ppd); +} + +/* + * 'colordUnregisterPrinter()' - Unregister profiles for a printer. + */ + +void +colordUnregisterPrinter(cupsd_printer_t *p) /* I - printer */ +{ + char device_id[1024]; /* Device ID as understood by colord */ + + /* + * Ensure we have a DBus connection + */ + + colordStart(); + + /* + * Just delete the device itself, and leave the profiles registered + */ + + snprintf(device_id, sizeof(device_id), "cups-%s", p->name); + colordDeleteDevice(device_id); +} + +#endif /* HAVE_DBUS */ + +/* + * End of "$Id$". + */ diff --git a/scheduler/colord.h b/scheduler/colord.h new file mode 100644 index 0000000..262b695 --- /dev/null +++ b/scheduler/colord.h @@ -0,0 +1,22 @@ +/* + * "$Id$" + * + * colord integration for the CUPS scheduler. + * + * Copyright 2011, Red Hat. + * + * These coded instructions, statements, and computer programs are the + * property of Apple Inc. and are protected by Federal copyright + * law. Distribution and use rights are outlined in the file "LICENSE.txt" + * which should have been included with this file. If this file is + * file is missing or damaged, see the license at "http://www.cups.org/". + */ + +void colordRegisterPrinter(cupsd_printer_t *p); +void colordUnregisterPrinter(cupsd_printer_t *p); +void colordStart(void); +void colordStop(void); + +/* + * End of "$Id$". + */ diff --git a/scheduler/ipp.c b/scheduler/ipp.c index 6ba8339..55b7ed3 100644 --- a/scheduler/ipp.c +++ b/scheduler/ipp.c @@ -2962,17 +2962,23 @@ add_printer(cupsd_client_t *con, /* I - Client connection */ cupsdSetPrinterReasons(printer, "none"); -#ifdef __APPLE__ /* * (Re)register color profiles... */ if (!RunUser) { + cupsdCmsRegisterPrinter(printer); +#ifdef __APPLE__ + /* + * FIXME: ideally the ColorSync stuff would be moved to colorsync.c + * and the colorsyncRegisterProfiles() would be called from + * cupsdCmsRegisterPrinter() in printers.c + */ apple_unregister_profiles(printer); apple_register_profiles(printer); - } #endif /* __APPLE__ */ + } } /* @@ -7052,11 +7058,17 @@ delete_printer(cupsd_client_t *con, /* I - Client connection */ snprintf(filename, sizeof(filename), "%s/%s.pwg3", CacheDir, printer->name); unlink(filename); -#ifdef __APPLE__ /* * Unregister color profiles... */ + cupsdCmsUnregisterPrinter(printer); +#ifdef __APPLE__ + /* + * FIXME: ideally the ColorSync stuff would be moved to colorsync.c + * and the colorsyncUnregisterPrinter() would be called from + * cupsdCmsUnregisterPrinter() in printers.c + */ apple_unregister_profiles(printer); #endif /* __APPLE__ */ diff --git a/scheduler/printers.c b/scheduler/printers.c index aedae5d..0947aeb 100644 --- a/scheduler/printers.c +++ b/scheduler/printers.c @@ -80,6 +80,9 @@ # include #endif /* HAVE_SYS_VFS_H */ +#ifdef HAVE_DBUS +# include "colord.h" +#endif /* HAVE_DBUS */ /* * Local functions... @@ -740,6 +743,53 @@ cupsdDeleteAllPrinters(void) } } +/* + * 'cupsdCmsRegisterPrinter()' - Registers a printer and profiles with the CMS + */ + +void +cupsdCmsRegisterPrinter(cupsd_printer_t *p) +{ +#if defined(HAVE_DBUS) + colordRegisterPrinter(p); +#endif /* defined(HAVE_DBUS) */ +} + +/* + * 'cupsdCmsUnregisterPrinter()' - Unregisters a printer and profiles with the CMS + */ + +void +cupsdCmsUnregisterPrinter(cupsd_printer_t *p) +{ +#if defined(HAVE_DBUS) + colordUnregisterPrinter(p); +#endif /* defined(HAVE_DBUS) */ +} + +/* + * 'cupsdCmsStart()' - Starts the CMS + */ + +void +cupsdCmsStart(void) +{ +#if defined(HAVE_DBUS) + colordStart(); +#endif /* defined(HAVE_DBUS) */ +} + +/* + * 'cupsdCmsStop()' - Stops the CMS + */ + +void +cupsdCmsStop(void) +{ +#if defined(HAVE_DBUS) + colordStop(); +#endif /* defined(HAVE_DBUS) */ +} /* * 'cupsdDeletePrinter()' - Delete a printer from the system. @@ -780,6 +830,12 @@ cupsdDeletePrinter( "Job stopped."); /* + * Unregister profiles... + */ + + cupsdCmsUnregisterPrinter(p); + + /* * If this printer is the next for browsing, point to the next one... */ @@ -1488,6 +1544,12 @@ cupsdRenamePrinter( p->prefiltertype = mimeAddType(MimeDatabase, "prefilter", name); /* + * Unregister profiles... + */ + + cupsdCmsUnregisterPrinter(p); + + /* * Rename the printer... */ @@ -2700,6 +2762,13 @@ cupsdSetPrinterAttrs(cupsd_printer_t *p)/* I - Printer to setup */ #endif /* __sgi */ /* + * Re-register profiles... + */ + + cupsdCmsUnregisterPrinter(p); + cupsdCmsRegisterPrinter(p); + + /* * Let the browse protocols reflect the change */ diff --git a/scheduler/printers.h b/scheduler/printers.h index 3afc88d..85d2c88 100644 --- a/scheduler/printers.h +++ b/scheduler/printers.h @@ -175,6 +175,10 @@ extern const char *cupsdValidateDest(const char *uri, cups_ptype_t *dtype, cupsd_printer_t **printer); extern void cupsdWritePrintcap(void); +extern void cupsdCmsRegisterPrinter(cupsd_printer_t *p); +extern void cupsdCmsUnregisterPrinter(cupsd_printer_t *p); +extern void cupsdCmsStart(void); +extern void cupsdCmsStop(void); /*