2 Unix SMB/CIFS implementation.
3 Main SMB server routines
4 Copyright (C) Andrew Tridgell 1992-1998
5 Copyright (C) Martin Pool 2002
6 Copyright (C) Jelmer Vernooij 2002-2003
7 Copyright (C) Volker Lendecke 1993-2007
8 Copyright (C) Jeremy Allison 1993-2007
10 This program is free software; you can redistribute it and/or modify
11 it under the terms of the GNU General Public License as published by
12 the Free Software Foundation; either version 3 of the License, or
13 (at your option) any later version.
15 This program is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 GNU General Public License for more details.
20 You should have received a copy of the GNU General Public License
21 along with this program. If not, see <http://www.gnu.org/licenses/>.
25 #include "smbd/smbd.h"
26 #include "smbd/globals.h"
27 #include "nt_printing.h"
28 #include "printing/pcap.h"
29 #include "printing/printer_list.h"
30 #include "printing/load.h"
33 #include "lib/param/loadparm.h"
36 * The persistent pcap cache is populated by the background print process. Per
37 * client smbds should only reload their printer share inventories if this
38 * information has changed. Use reload_last_pcap_time to detect this.
40 static time_t reload_last_pcap_time
= 0;
42 bool snum_is_shared_printer(int snum
)
44 return (lp_browseable(snum
) && lp_snum_ok(snum
) && lp_printable(snum
));
48 * @brief Purge stale printer shares and reload from pre-populated pcap cache.
50 * This function should normally only be called as a callback on a successful
51 * pcap_cache_reload(), or on client enumeration.
53 void delete_and_reload_printers(void)
60 time_t pcap_last_update
;
61 TALLOC_CTX
*frame
= NULL
;
62 const struct loadparm_substitution
*lp_sub
=
63 loadparm_s3_global_substitution();
65 if (!lp_load_printers()) {
66 DBG_DEBUG("skipping printer reload: disabled\n");
70 frame
= talloc_stackframe();
71 ok
= pcap_cache_loaded(&pcap_last_update
);
73 DEBUG(1, ("pcap cache not loaded\n"));
78 if (reload_last_pcap_time
== pcap_last_update
) {
79 DEBUG(5, ("skipping printer reload, already up to date.\n"));
83 reload_last_pcap_time
= pcap_last_update
;
85 /* Get pcap printers updated */
88 n_services
= lp_numservices();
89 pnum
= lp_servicenumber(PRINTERS_NAME
);
91 DEBUG(10, ("reloading printer services from pcap cache\n"));
94 * Add default config for printers added to smb.conf file and remove
97 for (snum
= 0; snum
< n_services
; snum
++) {
98 /* avoid removing PRINTERS_NAME */
103 /* skip no-printer services */
104 if (!snum_is_shared_printer(snum
)) {
108 pname
= lp_printername(frame
, lp_sub
, snum
);
110 /* check printer, but avoid removing non-autoloaded printers */
111 if (lp_autoloaded(snum
) &&
112 !printer_list_printername_exists(pname
)) {
113 lp_killservice(snum
);
117 /* Make sure deleted printers are gone */
123 /****************************************************************************
124 Reload the services file.
125 **************************************************************************/
127 bool reload_services(struct smbd_server_connection
*sconn
,
128 bool (*snumused
) (struct smbd_server_connection
*, int),
131 const struct loadparm_substitution
*lp_sub
=
132 loadparm_s3_global_substitution();
133 struct smbXsrv_connection
*xconn
= NULL
;
137 char *fname
= lp_next_configfile(talloc_tos(), lp_sub
);
138 if (file_exist(fname
) &&
139 !strcsequal(fname
, get_dyn_CONFIGFILE())) {
140 set_dyn_CONFIGFILE(fname
);
148 if (test
&& !lp_file_list_changed())
151 lp_killunused(sconn
, snumused
);
153 ret
= lp_load_with_shares(get_dyn_CONFIGFILE());
155 /* perhaps the config filename is now set */
157 reload_services(sconn
, snumused
, true);
164 if (sconn
!= NULL
&& sconn
->client
!= NULL
) {
165 xconn
= sconn
->client
->connections
;
167 for (;xconn
!= NULL
; xconn
= xconn
->next
) {
168 set_socket_options(xconn
->transport
.sock
, "SO_KEEPALIVE");
169 set_socket_options(xconn
->transport
.sock
, lp_socket_options());
172 mangle_reset_cache();