1 /**************************************************************************/
2 /*                                                                        */
3 /*       Copyright (c) Microsoft Corporation. All rights reserved.        */
4 /*                                                                        */
5 /*       This software is licensed under the Microsoft Software License   */
6 /*       Terms for Microsoft Azure RTOS. Full text of the license can be  */
7 /*       found in the LICENSE file at https://aka.ms/AzureRTOS_EULA       */
8 /*       and in the root directory of this software.                      */
9 /*                                                                        */
10 /**************************************************************************/
11 
12 
13 /**************************************************************************/
14 /**************************************************************************/
15 /**                                                                       */
16 /** USBX Component                                                        */
17 /**                                                                       */
18 /**   Printer Class                                                       */
19 /**                                                                       */
20 /**************************************************************************/
21 /**************************************************************************/
22 
23 
24 /* Include necessary system files.  */
25 
26 #define UX_SOURCE_CODE
27 
28 #include "ux_api.h"
29 #include "ux_host_class_printer.h"
30 #include "ux_host_stack.h"
31 
32 
33 /**************************************************************************/
34 /*                                                                        */
35 /*  FUNCTION                                               RELEASE        */
36 /*                                                                        */
37 /*    _ux_host_class_printer_configure                    PORTABLE C      */
38 /*                                                           6.1          */
39 /*  AUTHOR                                                                */
40 /*                                                                        */
41 /*    Chaoqiong Xiao, Microsoft Corporation                               */
42 /*                                                                        */
43 /*  DESCRIPTION                                                           */
44 /*                                                                        */
45 /*    This function calls the USBX stack to do a SET_CONFIGURATION to the */
46 /*    printer. Once the printer is configured, its interface will be      */
47 /*    activated. The bulk endpoints enumerated(1 IN, 1 OUT ).             */
48 /*                                                                        */
49 /*  INPUT                                                                 */
50 /*                                                                        */
51 /*    printer                               Pointer to printer class      */
52 /*                                                                        */
53 /*  OUTPUT                                                                */
54 /*                                                                        */
55 /*    Completion Status                                                   */
56 /*                                                                        */
57 /*  CALLS                                                                 */
58 /*                                                                        */
59 /*    _ux_host_stack_configuration_interface_get  Get interface           */
60 /*    _ux_host_stack_device_configuration_get     Get configuration       */
61 /*    _ux_host_stack_device_configuration_select  Select configuration    */
62 /*                                                                        */
63 /*  CALLED BY                                                             */
64 /*                                                                        */
65 /*    _ux_host_class_printer_activate       Printer class activate        */
66 /*                                                                        */
67 /*  RELEASE HISTORY                                                       */
68 /*                                                                        */
69 /*    DATE              NAME                      DESCRIPTION             */
70 /*                                                                        */
71 /*  05-19-2020     Chaoqiong Xiao           Initial Version 6.0           */
72 /*  09-30-2020     Chaoqiong Xiao           Modified comment(s),          */
73 /*                                            optimized based on compile  */
74 /*                                            definitions,                */
75 /*                                            resulting in version 6.1    */
76 /*                                                                        */
77 /**************************************************************************/
_ux_host_class_printer_configure(UX_HOST_CLASS_PRINTER * printer)78 UINT  _ux_host_class_printer_configure(UX_HOST_CLASS_PRINTER *printer)
79 {
80 
81 UINT                    status;
82 UX_CONFIGURATION        *configuration;
83 #if UX_MAX_DEVICES > 1
84 UX_DEVICE               *parent_device;
85 #endif
86 
87 
88     /* If the device has been configured already, we don't need to do it
89        again. */
90     if (printer -> ux_host_class_printer_device -> ux_device_state == UX_DEVICE_CONFIGURED)
91         return(UX_SUCCESS);
92 
93     /* A printer normally has one configuration. So retrieve the 1st configuration
94        only.  */
95     status =  _ux_host_stack_device_configuration_get(printer -> ux_host_class_printer_device, 0, &configuration);
96     if (status != UX_SUCCESS)
97     {
98 
99         /* Error trap. */
100         _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_CONFIGURATION_HANDLE_UNKNOWN);
101 
102         /* If trace is enabled, insert this event into the trace buffer.  */
103         UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_CONFIGURATION_HANDLE_UNKNOWN, printer -> ux_host_class_printer_device, 0, 0, UX_TRACE_ERRORS, 0, 0)
104 
105         return(UX_CONFIGURATION_HANDLE_UNKNOWN);
106     }
107 
108 #if UX_MAX_DEVICES > 1
109     /* Check the printer power source and check the parent power source for
110        incompatible connections.  */
111     if (printer -> ux_host_class_printer_device -> ux_device_power_source == UX_DEVICE_BUS_POWERED)
112     {
113 
114         /* Get parent device pointer.  */
115         parent_device =  printer -> ux_host_class_printer_device -> ux_device_parent;
116 
117         /* If the device is NULL, the parent is the root printer and we don't have to worry
118            if the parent is not the root printer, check for its power source.  */
119         if ((parent_device != UX_NULL) && (parent_device -> ux_device_power_source == UX_DEVICE_BUS_POWERED))
120         {
121 
122             /* Error trap. */
123             _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_CONNECTION_INCOMPATIBLE);
124 
125             /* If trace is enabled, insert this event into the trace buffer.  */
126             UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_CONNECTION_INCOMPATIBLE, printer, 0, 0, UX_TRACE_ERRORS, 0, 0)
127 
128             return(UX_CONNECTION_INCOMPATIBLE);
129         }
130     }
131 #endif
132 
133     /* We have the valid configuration. Ask the USBX stack to set this configuration.  */
134     status =  _ux_host_stack_device_configuration_select(configuration);
135     if (status != UX_SUCCESS)
136         return(status);
137 
138     /* If the operation went well, the printer default alternate setting for the printer interface is
139        active and the interrupt endpoint is now enabled. We have to memorize the first interface since
140        the interrupt endpoint is hooked to it. */
141     status =  _ux_host_stack_configuration_interface_get(configuration, 0, 0, &printer -> ux_host_class_printer_interface);
142     if (status != UX_SUCCESS)
143     {
144 
145         /* Store the instance in the interface container, this is for the USB stack
146            when it needs to invoke the class.  */
147         printer -> ux_host_class_printer_interface -> ux_interface_class_instance =  (VOID *) printer;
148     }
149 
150     /* Return completion status.  */
151     return(status);
152 }
153 
154